質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.92%

Java参照渡しについて

解決済

回答 3

投稿

  • 評価
  • クリップ 0
  • VIEW 290

kobe2018

score 21

以下のコードがいまいち理解できません。
自分なりの解釈を書いてみたので合っているか教えてください。

public class Main {
    public static void main(String[] args) {
        String s = "abc";
        method(s);
        System.out.println(s);
    }

    public static void method(String s) {
        s = s + "def";
    }

}

実行結果は以下です。

abc

methodメソッドではsに新しい値を入力していると思いました。
methodメソッドは以下のようなことをやっているという解釈であってますか。

s = new String(s + "def");

つまり参照先を新しくしている。
そのため、mainメソッドのsには関係がなくなる。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 3

+4

Javaは参照渡しではないため、新たなオブジェクトを仮引数に代入しても呼び出し元側の実引数は変更されません。Stringが不変とかそういうのは今回は全く関係無いです。

まず、Javaでは、Stringオブジェクトや文字列リテラルに対する+演算子(String Concatenation Operator +と言われる)は結合された新たなStringを生成します。一般的なJDKの実装では、StringBuilder(昔はStringBuffer)を用いて文字列を結合し、最後にStringに変換するという処理を行っています。つまり、s + "def"は新たな別のStringです。それを、仮引数sに代入していることになりますが、これは呼び出し元の実引数sとは別物です。methodメソッドにおいて、全てのsxに変えても同じ動きをすることからもそれはわかるでしょう。たまたま名前が同じなだけなので、ここは混同しないでください。そして、Javaでのメソッドの引数渡しは全て値渡しであるため、仮引数に新たな値を代入しても、実引数側には何も影響を与えないと言うことです。

これはStringが不変とかは関係ありません。例えば、可変であるStringBuilderであっても、結果は同じです。

public class Main {
    public static void main(String[] args) {
        var s = new StringBuilder("abc");
        method(s);
        System.out.println(s);
    }

    public static void method(StringBuilder s) {
        var x = new StringBuilder();
        x.append(s.toString());
        x.append("def");
        s = x;
    }
}

しかし、StringBuilderの場合は、次のように書くと結果が変わります。(Stringは不変であるため、このような書き方はできません。)

public class Main {
    public static void main(String[] args) {
        var s = new StringBuilder("abc");
        method(s);
        System.out.println(s);
    }

    public static void method(StringBuilder s) {
        s.append("def");
    }
}

こちらは"abcdfe"となるはずです。これはオブジェクトそのものを変更しているためです。Javaの参照型では、変数が持つのが参照値であり、それを値渡しで渡しているため、このようなことが可能です。このようなことができる値渡しを「参照の値渡し」(共有渡し、オブジェクト渡し、call by share、call by object、call by object-sharing、また、渡しではなく呼び、call byではなくpass byという表現もあり、今のところ定まった一般的な訳語がない)と呼ばれます。

では、C++やC#の参照渡しの場合はどうなるのかというと、実際に見てみましょう。(C++だとオブジェクトの持ち方がJavaと違うので似た仕組みになっているC#を使います。)

public class Test {
    public static void Main() {
        var s = "abc";
        method(ref s);
        System.Console.WriteLine(s);
    }

    public static void method(ref string s) {
        s = s + "def";
    }
}

stringはJavaのString相当で参照型の不変な文字列オブジェクトで、refは参照渡しであることを示しています。参照渡しであるため、仮引数への代入によって実引数も変更され、結果は"abcdef"となります。逆に、参照渡しではなく、Javaと同じく値渡しにした場合を見てみましょう。

public class Test {
    public static void Main() {
        var s = "abc";
        method(s);
        System.Console.WriteLine(s);
    }

    public static void method(string s) {
        s = s + "def";
    }
}

Javaと同じく"abc"になります。

各コードはpaiza.ioで確認してみてください。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

checkベストアンサー

0

methodメソッドは以下のようなことをやっているという解釈であってますか。

はい、そのとおりです。


JavaのStringイミュータブルです。つまり、一度作られたインスタンスの中身を変える方法は存在しません

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2020/07/18 23:46

    そうですね。すみません。
    s1の参照先に格納されている"a"に訂正します。

    キャンセル

  • 2020/07/18 23:46

    ありがとうございます。

    キャンセル

  • 2020/07/18 23:47

    > s1の参照先に格納されている"a"に訂正します。

    違います。s1のインスタンスと同じ内容で、新たに作ったインスタンスです。

    キャンセル

0

   s = s + "def";

上記操作を行いたい場合は StringBuilder 等で実装しましょう。
メモリ効率もそのほうが有効です。

Stringのインスタンスは不変ですし、 
✕「引数に当てた変数の値を変更する」
◯「引数の変数の値を変更する」なので「引数に代入したもとの変数」は変わりません。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.92%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る