【最大10000ポイント】当たる!!質問投稿キャンペーン!

お世話になります。

下記のソースを実行すると

public class ExecMain {

public static void main(String[] args) {
ObjectLife obj = new ObjectLife();
ObjectLife obj2 = obj;
obj = null;
System.out.println(obj);
System.out.println(obj2);
}

static class ObjectLife {
public ObjectLife() {
create();
}
public void create() {
System.out.println("create calling");
}
public void destroy() {
System.out.println("destroy calling");
this = null;
}
}
}
--------------------
create calling
null
objectLife.ExecMain$ObjectLife@1b90b39

という実行結果になりました。
「objectLife.ExecMain$ObjectLife@1b90b39」
がnullにならないのはどうしてなのでしょう?

また
this=null;
で自分自身のインスタンスをnullにしようとおもったのですが
コンパイルはとおるのですが実行してもなにもおきていないようです。

これはどのように解釈すればよいのでしょうか?

以上よろしくお願いいたします。

A 回答 (2件)

Javaのオブジェクトの変数は「参照」を保存する為の入れ物です(決して、オブジェクトそのものを保持する訳ではない。

ましてや、変数自体の「参照」でもない)

例えば、、、

String s1=new String("a");  //s1→「001」

とした場合、新しい文字列オブジェクトがメモリに展開されます(String s1="a"; となると微妙に意味合いが異なるので敢えて new で新しい文字列オブジェクトを作った場合で話を進めます)
そして、展開された文字列オブジェクトがメモリの何処にあるのかを示す番地が「参照」という事です。
上記の場合、例えとして「001」という番地(参照)だったとしましょう。
上記の記述でしたらこの後で、この番地を s1 が保持する事になります(s1 には「001」という番地情報が入る)
次に、、、

String s2=s1;   //s2→「001」

とした場合、s1 に保持されている番地(「001」)が s2 にコピーされます。
ですから、この後に、、、

System.out.println(s1);  //s1→「001」
System.out.println(s2);  //s2→「001」

とすると、どちらも同じメモリ位置にある同じオブジェクトを表示する事になり同じ出力となります。
a
a

続いて、、、

s1=new String("b");   //s1→「002」

とすると、メモリに新たな文字列オブジェクトが生成され、その番地が文字列型変数 s1 の前の保持していた番地情報と入れ替わります。
例えば、新しい文字列オブジェクトの位置情報が「002」とすると「001」→「002」に変化するというわけですね。
まとめてみると、、、

String s1=new String("a");  //s1→「001」
String s2=s1;          //s1,s2→「001」
s1=new String("b");      //s1→「002」,s2→「001」

このように変化しますので、この後に出力させると、、、

b
a

という事になります。
    • good
    • 0

ObjectLife obj = new ObjectLife();


ObjectLife obj2 = obj;

1行目は new で作られたオブジェクトの「参照アドレス」
を obj という変数に格納しているわけですよね。
そして、2行目で obj2 に obj に格納されている「参照
アドレス」の値をコピーして渡している。
、、、と考えれば良いのではないでしょうか。

要するに、

obj2 → obj → 「new」で作られたオブジェクトの参照

という直列の関係ではなくて、

obj →
      「new」で作られたオブジェクトの参照
obj2 →

という並列関係で捉えれば分かり易いかもしれません。
ですから、obj に null を代入したところで obj2 が
持っているオブジェクトの参照値が変わる訳では無い、、
という事では無いでしょうか。
そして、この後で obj2 にも null を代入したとしても
直ぐにオブジェクト自体が無くなるというわけじゃなくて、
ガベージコレクションの候補として登録され、VM が
ガベージコレクションを起動させたときに始めてそこで
オブジェクトが破棄されるという流れだと思います。

>this=null; で自分自身のインスタンスをnullにしよう
>とおもったのですが、コンパイルはとおるのですが実行
>してもなにもおきていないようです。

あれ?私のコンパイラではコンパイル通りませんでしたよ。

「final 変数 this に値を代入する事はできません」

この回答への補足

アドバイスありがとうございます。
なんとなくわかったようなまだなぞのようなという感じです。

public static void main(String[] args) {
String s1= "a";
String s2= s1;
s1 ="b";
System.out.println(s1);
System.out.println(s2);
}

上記のソースを実行した場合
b
a
となってしまいます。

これはどのように解釈すればよいのでしょうか?
またこれを


とするようにするにはどのようにしたらよいのでしょうか?

補足日時:2003/05/17 15:46
    • good
    • 0

お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!


人気Q&Aランキング