Как работает "object.new"? (Есть ли в Java оператор .new?)

Я столкнулся с этим кодом сегодня, читая Ускоренный GWT (Gupta) - стр. 151.

public static void getListOfBooks(String category, BookStore bookStore) {
    serviceInstance.getBooks(category, bookStore.new BookListUpdaterCallback());
}
public static void storeOrder(List books, String userName, BookStore bookStore) {
    serviceInstance.storeOrder(books, userName,    bookStore.new StoreOrderCallback());
}

Что там делают новые операторы? Я никогда не видел такой синтаксис, может ли кто-нибудь объяснить?

Кто-нибудь знает, где найти это в спецификации java?

Ответ 1

Они внутренние (вложенные нестатические) классы:

public class Outer {
  public class Inner { public void foo() { ... } }
}

Вы можете сделать:

Outer outer = new Outer();
outer.new Inner().foo();

или просто:

new Outer().new Inner().foo();

Причиной этого является то, что Inner имеет ссылку на конкретный экземпляр внешнего класса. Позвольте мне привести более подробный пример этого:

public class Outer {
  private final String message;

  Outer(String message) {
    this.message = message;
  }

  public class Inner {
    private final String message;

    public Inner(String message) {
       this.message = message;
    }

    public void foo() {
      System.out.printf("%s %s%n", Outer.this.message, message);
    }
  }
}

и запустите:

new Outer("Hello").new Inner("World").foo();

Выходы:

Hello World

Примечание. вложенные классы могут быть static тоже. Если это так, у них нет неявной ссылки this для внешнего класса:

public class Outer {
  public static class Nested {
    public void foo() { System.out.println("Foo"); }
  }
}

new Outer.Nested.foo();

Чаще всего статические вложенные классы private, поскольку они, как правило, являются деталями реализации и опрятным способом инкапсуляции части проблемы без загрязнения публичного пространства имен.

Ответ 3

Я еще не видел этот синтаксис, но я думаю, что он создаст внутренний класс BookStore.