2011-09-19 30 views
17

En final variable passed to anonymous class via constructor, Jon Skeet mencionó que las variables se pasan a la instancia de clase anónima a través de un constructor autogenerado. ¿Por qué no iba a ser capaz de ver el constructor utilizando la reflexión en ese caso:Pasar variables finales a clases anónimas

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); 

}

La salida es:

100 
constructor : A$1() 

Respuesta

16

Esto es lo que su programa imprime en mi sistema:

100 
constructor : A$1() 

Así que el constructor está ahí . Sin embargo, no tiene parámetros. Desde el desmontaje, lo que sucede es que el compilador se da cuenta de que no necesita pasar x a run() ya que su valor es conocido en tiempo de compilación.

si cambio el código de este modo:

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); 
    } 

} 

El constructor que se genera es ahora:

constructor : A$1(int) 

El único argumento es el valor de x.

27

En este caso, es porque 100 es una constante. Eso se cuece en tu clase.

Si cambia x a ser:

final int x = args.length; 

... entonces usted verá Test$1(int) en la salida. (Esto es a pesar de no haber sido declarado explícitamente Y sí, capturando más variables añade parámetros al constructor..)

+1

@Bohemian: Dado que sé el origen de la pregunta, creo que es :) –

Cuestiones relacionadas