2011-09-19 12 views
17

W książce final variable passed to anonymous class via constructor Jon Skeet wspomniał, że zmienne są przekazywane do anonimowej instancji klasy przez wygenerowany automatycznie konstruktor. Dlaczego miałbym nie być w stanie zobaczyć konstruktora przy użyciu odbicia w tym przypadku:Przekazywanie zmiennych końcowych do anonimowych klas

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

}

Wyjście jest:

100 
constructor : A$1() 

Odpowiedz

16

Oto co program wypisuje na moim systemie:

100 
constructor : A$1() 

Więc konstruktor jest tam . Jednak jest bez parametrów. Patrząc na demontaż, kompilator domyśla się, że nie musi on przekazywać x do run(), ponieważ jego wartość jest znana podczas kompilacji.

Jeśli zmienić kod tak:

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

} 

konstruktora, który pobiera generowany jest teraz:

constructor : A$1(int) 

Jedynym argumentem jest wartość x.

27

W tym przypadku, to dlatego, że 100 jest stała. To wpakuje się w twoją klasę.

Jeżeli zmienisz x być:

final int x = args.length; 

... wtedy zobaczysz Test$1(int) na wyjściu. (Jest to, mimo że nie to jest jawnie zadeklarowane i tak, zdobywając kolejne zmienne dodaje parametry do konstruktora.).

+1

@Bohemian: Biorąc pod uwagę, że znam pochodzenie pytania, myślę, że to jest :) –

Powiązane problemy