• ,

Уровень 23. Ответы на вопросы к собеседованию по теме уровня.

1. Какие бывают внутренние классы?
Вложенные классы делятся на два вида: статические и не статические.
  • Вложенные классы, объявленные как статические называются вложенными статическими (static nested classes).
  • Вложенные не статические классы называются внутренними (inner classes).

2. Во что компилируется анонимный внутренний класс?
Во внутренний не статический класс

3. Зачем использовать ключевое слово final при создании анонимных классов?
Если определяется анонимный внутренний класс и ему нужно при этом использовать объекты, определенные вне этого внутреннего класса, компилятор требует, чтобы переданные на них ссылки объявлялись неизменными (final). Без такого объявления вы получите сообщение об ошибке при компиляции программы.

4. Как правильно создать объект внутреннего класса?
Внутренние (не статические) классы, как переменные и методы связаны с объектом внешнего класса. Внутренние классы так же имеют прямой доступ к полям внешнего класса. Такие классы не могут содержать в себе статические методы и поля. Внутренние классы не могут существовать без экземпляра внешнего. Для создания объекта:
Outer outer = new Outer();
Innter inner = outer.new Inner();


5. Как правильно создать объект вложенного класса?
Синтаксис создания объекта вложенного класса:
OuterClass.StaticNestedClass nestedObject = new OuterClass.StaticNestedClass();


6. Можно ли создавать статические методы/переменные во внутреннем классе?
Статические методы/переменные объявлять во внутреннем классе (не вложенном) нельзя.
Внутренние (не статические) классы, как переменные и методы связаны с объектом внешнего класса. Такие классы не могут содержать в себе статические методы и поля.

7. Назовите три любых внутренних класса?
  1. private static class Holder —вложенный класс HashMap из java.util.
  2. В интерфейсе Map есть interface Entry<K,V>, который опять же в HashMap и реализуется в другом вложенном классе static class Entry<K,V> implements Map.Entry<K,V>.
  3. private static class IntegerCache в классе Integer .

8. Как внутренние классы решают проблему множественного наследования в Java?
Т.к. множественное наследование классов в Java запрещено, эту проблему решают с помощью внутренних классов: в нужном нам классе мы объявляем внутренний класс и наследуем его от требуемого класса. Пример:

class Tiger extends Cat
{

 public void tigerRun()
 {
  .....
 }

public void startTiger()
 {
  TigerThread thread = new TigerThread();
  thread.start();
 }

 class TigerThread extends Thread
 {
  public void run()
  {
   tigerRun();
  } 
 }
}


9. Чем отличаются анонимные классы, созданные на основе интерфейса и на основе класса?
Анонимный класс согласно JLS 15.9.5 представляют собой выражение, в котором объявление нового класса и его инициализация совмещены:

При объявлении класса будет создан новый класс, производный от указанного класса при использовании в качестве базы другого класса, или реализующий интерфейс при использовании в качестве базы интерфейса.
При инициализации будет создан новый объект и на него будет возвращения ссылка: анонимный класс является конкретным.
Таким образом, единственное отличие в анонимных классах, созданных на основе интерфейса и класса, заключается в количестве абстракных методов, которые необходимо реализовать.

Инициализации анонимного класса на основе интерфейса потребует реализации каждого метода, в то время как при инициализации анонимного класса на основе абстрактного/конкретного класса позволит изменить поведение за счет перегрузки методов и потребует реализации абстракных методов.

10. Можно ли создать анонимный статический вложенный класс?
Нет, статической становится только переменная, но не класс.

8 комментариев

Joysi
1. Какие бывают внутренние классы?
Вложенные классы делятся на два вида: статические и не статические.
Вложенные классы, объявленные как статические называются вложенными статическими (static nested classes).
Вложенные не статические классы называются внутренними (inner classes).

Я бы добавил.
Помимо объявления вложенного класса внутри внешнего по отношению к нему, возможно также определить внутренний класс непосредственно в блоке кода (например, в теле метода и блоках инициализации). При этом, можно в блоке:
— объявить класс и в дальнейшем его использовать (local inner classes),
— одновременно объявить и создать класс динамически (anonymous inner classes).
zor07
Спасибо за замечание )
TtTt
Надо бы поправить:
1. Вложенные не статические классы называются внутренними (inner classes)
2. Назовите три любых внутренних класса?
private static class Holder —вложенный класс HashMap из java.util.

Сначала внутренний класс классифицирован как не статический. Затем в примере приведены статические классы. Наверное в примере должны быть нестатические классы :)
Core
  • Core
  • 0
  • Комментарий отредактирован 2016-12-01 17:04:33 пользователем Core
3. Зачем использовать ключевое слово final при создании анонимных классов?
Если определяется анонимный внутренний класс и ему нужно при этом использовать объекты, определенные вне этого внутреннего класса, компилятор требует, чтобы переданные на них ссылки объявлялись неизменными (final). Без такого объявления вы получите сообщение об ошибке при компиляции программы.
Вот этот ответ некорректен, на мой взягляд. Анонимный внутренний класс, как и локальный класс вполне может обратиться ко внешним переменным, определённым вне этого класса и не имеющим модификатор final. Например, если эта переменная — поле обрамляющего класса. А вот если нужно обратиться в локальном классе к локальным переменным внешнего класса или к аргументу метода, в котором определён локальный класс — тогда да, нужен модификатор final. Ниже код, который скомпилируется даже при отсутствии final у fieldOfOuterClass

<code>public class OuterClass {
    int fieldOfOuterClass = 222;
    public void someMethod(final int arg,int i){
        final int localVariableOfOuterClass = 100;
        class LocalClass{
            public void printArg(){
                System.out.println("LocalClass.printArg(field--> "+fieldOfOuterClass+" | "+arg+" <--arg),"+" localPatametr--> "+localVariableOfOuterClass);
            }
        }
        new LocalClass().printArg();
    }
    public static void main(String[] args) {
        new OuterClass().someMethod(321,123);
    }
}</code>

Я бы так ответил: модификатор final нужно использовать при возникновении необходимсоти обратиться из локального/анонимного внутреннего класса к локальным переменным внешнего класса или к аргументам метода, в котором этот локальный/анонимный внутренний класс реализован.

Могу ошибаться, но эксперименты с кодом заставили меня придти к такому выводу.
Rosewelt
3. Зачем использовать ключевое слово final при создании анонимных классов?
Если определяется анонимный внутренний класс и ему нужно при этом использовать объекты, определенные вне этого внутреннего класса, компилятор требует, чтобы переданные на них ссылки объявлялись неизменными (final). Без такого объявления вы получите сообщение об ошибке при компиляции программы.
However, starting in Java SE 8, a local class can access local variables and parameters of the enclosing block that are final or effectively final. A variable or parameter whose value is never changed after it is initialized is effectively final.
fatfaggy
but you still need to use «final» keyword anyway :)
Rosewelt
No, this example compiles and works great:
public class Main {

    public static void main(String[] args) {
        new Main().method1();
    }

    public void method1(){
        int a = 4;
        // a = 4; compile error on line 12 ('a' is not final or effectively final)
        class Local {
            public void printA(){
                System.out.println(a); // line 12
            }
        }

        new Local().printA();
    }
}
fatfaggy
  • fatfaggy
  • 0
  • Комментарий отредактирован 2018-01-13 16:05:48 пользователем fatfaggy
try something like this:
<code>public void printA(){
    a++;
    System.out.println(a); // line 12
}</code>

and the question you quoted was about anonymous classes, not about local ones. but this error will be in both of them, yes :)
Только зарегистрированные и авторизованные пользователи могут оставлять комментарии.