2011-09-19 3 views
17

final variable passed to anonymous class via constructorで、Jon Skeetは変数が自動生成コンストラクタを介して匿名クラスインスタンスに渡されると述べています。なぜその場合、反射を使用してコンストラクタを見ることができないであろう。最終変数を匿名クラスに渡す

public static void main(String... args) throws InterruptedException { 
final int x = 100; 
new Thread() { 
    public void run() { 
     System.out.println(x);  
     for (Constructor<?> cons : this.getClass() 
       .getDeclaredConstructors()) { 
      StringBuilder str = new StringBuilder(); 
      str.append("constructor : ").append(cons.getName()) 
        .append("("); 
      for (Class<?> param : cons.getParameterTypes()) { 
       str.append(param.getSimpleName()).append(", "); 
      } 
      if (str.charAt(str.length() - 1) == ' ') { 
       str.replace(str.length() - 2, str.length(), ")"); 
      } else 
       str.append(')'); 
      System.out.println(str); 
     } 
    } 

}.start(); 
Thread.sleep(2000); 

}

出力である:この場合

100 
constructor : A$1() 

答えて

16

は、あなたのプログラムは、私のシステムで出力しますものです:

100 
constructor : A$1() 

ので、コンストラクタがあります。しかし、それは無パラメータです。逆アセンブルを見ると、コンパイラはコンパイル時にその値がわかっているので、xrun()に渡す必要はないと判断します。

私はそうのようにコードを変更した場合:

public class A { 

    public static void test(final int x) throws InterruptedException { 
     new Thread() { 
      public void run() { 
       System.out.println(x); 
       for (Constructor<?> cons : this.getClass() 
         .getDeclaredConstructors()) { 
        StringBuilder str = new StringBuilder(); 
        str.append("constructor : ").append(cons.getName()) 
          .append("("); 
        for (Class<?> param : cons.getParameterTypes()) { 
         str.append(param.getSimpleName()).append(", "); 
        } 
        if (str.charAt(str.length() - 1) == ' ') { 
         str.replace(str.length() - 2, str.length(), ")"); 
        } else 
         str.append(')'); 
        System.out.println(str); 
       } 
      } 

     }.start(); 
     Thread.sleep(2000); 
     } 

    public static void main(String[] args) throws InterruptedException { 
     test(100); 
    } 

} 

生成されますコンストラクタは以下のようになります。

constructor : A$1(int) 

唯一の引数はxの値です。

27

100は一定であるので、それはです。それはあなたのクラスに焼かれる。

あなたがするxを変更する場合:

final int x = args.length; 

...そしてあなたは出力にTest$1(int)が表示されます。 (これは、それにもかかわらず、明示的に宣言されていないはい、複数の変数コンストラクタにパラメータを追加取得する。。)ここで

+1

@ボヘミアン:私が質問の起源を知っていることを考えると、私はそれが:) –

関連する問題