Static nested classes in java
Объясните кто знает что за модификатора доступа (package private)? Чем он отличается от обычного private? Могу предположить что автор имел ввиду default access modifier, поправьте если не прав.
Продолжение про нестатические вложенные (анонимные) классы (2/3 часть): https://javarush.com/groups/posts/2193-anonimnihe-klassih Продолжение про статические вложенные классы (3/3 часть): https://javarush.com/groups/posts/2183-staticheskie-vlozhennihe-klassih
Зачем давать не актуальную информацию? Внутренний класс МОЖЕТ содержать статические переменные и методы но с JAVA 16.
В шестом пункте написано » Если у внутреннего класса нет модификатора доступа (package private), объекты внутреннего класса можно создавать внутри «внешнего» класса;» , а если есть, то нельзя?
Так же protected работает и для внутренних классов. Объекты protected внутреннего класса можно создавать: внутри «внешнего» класса; в его классах-наследниках; в тех классах, которые находятся в том же пакете. Может кто подсказать, что за хрень творится с этим модификатором доступа? Почему я не могу создать экземпляр внутреннего класса в классе-наследнике (наследование от внешнего класса, так как от внутреннего protected компилятор не дает наследоваться) в другом пэкадже? Хотя в статье говориться, что я могу это сделать.
Вообще не понимаю что тут написано 🤷🏽♂️ «Объект внутреннего класса нельзя создать в статическом методе «внешнего» класса. Это объясняется особенностями устройства внутренних классов. У внутреннего класса могут быть конструкторы с параметрами или только конструктор по умолчанию. Но независимо от этого, когда мы создаем объект внутреннего класса, в него незаметно передается ссылка на объект «внешнего» класса. Ведь наличие такого объекта — обязательное условие. Иначе мы не сможем создавать объекты внутреннего класса. Но если метод внешнего класса статический, значит, объект внешнего класса может вообще не существовать! А значит, логика работы внутреннего класса будет нарушена. В такой ситуации компилятор выбросит ошибку:
public static Seat createSeat() < //Bicycle.this cannot be referenced from a static context return new Seat(); >
Объект внутреннего класса нельзя создать в статическом методе «внешнего» класса. Может быть его нельзя создать не потому что есть или нет объекта внешнего класса, который можно создать по дефолту, а потому что в статических методах нельзя использовать нестатические переменные? public class Main < public static void main(String[] args) < Outer obj = new Outer(); Outer.Inner obj1 = obj.getObj(); System.out.println(obj1.a); >> class Outer < static class Inner < int a = 5; >static Inner getObj() < return new Inner(); >>
Статические вложенные классы
Привет! Мы продолжаем изучать тему вложенных классов (nested classes) в Java. На прошлом занятии мы поговорили о нестатических внутренних классах (non-static nested classes) или, как их еще называют, внутренних классах. Сегодня перейдем к другой группе и рассмотрим подробнее статические вложенные классы (static nested classes). Чем они отличаются от остальных групп? При объявлении такого класса мы используем уже знакомое тебе ключевое слово static:
public class Boeing737 < private int manufactureYear; private static int maxPassengersCount = 300; public Boeing737(int manufactureYear) < this.manufactureYear = manufactureYear; >public int getManufactureYear() < return manufactureYear; >public static class Drawing < public static int getMaxPassengersCount() < return maxPassengersCount; >> >
В этом примере у нас есть внешний класс Boeing737 , который создает самолет этой модели. А у него — конструктор с одним параметром: годом выпуска ( int manufactureYear ). Также есть одна статическая переменная int maxPassengersCount — максимальное число пассажиров. Оно будет одинаковым у всех самолетов одной модели, так что нам достаточно одного экземпляра. Кроме того, у него есть статический внутренний класс Drawing — чертеж самолета. В этом классе мы можем инкапсулировать всю служебную информацию о самолете. В нашем примере для простоты мы ограничили ее годом выпуска, но она может содержать много другой информации. Как мы и говорили в прошлой лекции, создание такого вложенного класса повышает инкапсуляцию и способствует более реалистичной абстракции. В чем же отличие между статическим и нестатическим вложенными классами? 1. Объект статического класса Drawing не хранит ссылку на конкретный экземпляр внешнего класса. Вспомни пример из прошлой лекции с велосипедом:
public class Bicycle < private String model; private int mawWeight; public Bicycle(String model, int mawWeight) < this.model = model; this.mawWeight = mawWeight; >public void start() < System.out.println("Поехали!"); >public class SteeringWheel < public void right() < System.out.println("Руль вправо!"); >public void left() < System.out.println("Руль влево!"); >> >
Там мы говорили о том, что в каждый экземпляр внутреннего класса SteeringWheel (руль) незаметно для нас передается ссылка на объект внешнего класса Bicycle (велосипед). Без объекта внешнего класса объект внутреннего просто не мог существовать. Для статических вложенных классов это не так. Объект статического вложенного класса вполне может существовать сам по себе. В этом плане статические классы более «независимы», чем нестатические. Единственный момент — при создании такого объекта нужно указывать название внешнего класса:
Почему мы сделали класс Drawing статическим, а в прошлой лекции класс Seat (сиденье велосипеда) был нестатическим? Как и в прошлый раз, давай добавим немного «философии» для понимания примера 🙂 В отличие от сиденья велосипеда, сущность чертежа не привязана так жестко к сущности самолета. Отдельный объект сиденья, без велосипеда, чаще всего будет бессмысленным (хотя и не всегда — мы говорили об этом на прошлом занятии). Сущность чертежа имеет смысл сама по себе. Например, он может пригодиться инженерам, планирующим ремонт самолета. Сам самолет для планирования им не нужен, и может находиться где угодно — достаточно просто чертежа. Кроме того, для всех самолетов одной модели чертеж все равно будет одинаковым, так что такой жесткой связи, как у сиденья с велосипедом, нет. Поэтому и ссылка на конкретный объект самолета объекту Drawing не нужна. 2. Разный доступ к переменным и методам внешнего класса. Статический вложенный класс может обращаться только к статическим полям внешнего класса. В нашем примере в классе Drawing есть метод getMaxPassengersCount() , который возвращает значение статической переменной maxPassengersCount из внешнего класса. Однако мы не можем создать метод getManufactureYear() в Drawing для возврата значения manufactureYear . Ведь переменная manufactureYear — нестатическая, а значит, должна принадлежать конкретному экземпляру Boeing737 . А как мы уже выяснили, в случае со статическими вложенными классами объект внешнего класса запросто может отсутствовать. Отсюда и ограничение 🙂 При этом неважно, какой модификатор доступа имеет статическая переменная во внешнем классе. Даже если это private , доступ из статического вложенного класса все равно будет. Все вышесказанное касается не только доступа к статическим переменным, но и к статическим методам. ВАЖНО! Слово static в объявлении внутреннего класса не означает, что можно создать всего один объект. Не путай объекты с переменными. Если мы говорим о статических переменных — да, статическая переменная класса, например, maxPassangersCount , существует в единственном экземпляре. Но применительно ко вложенному классу static означает лишь то, что его объекты не содержат ссылок на объекты внешнего класса. А самих объектов мы можем создать сколько угодно:
public class Boeing737 < private int manufactureYear; private static int maxPassengersCount = 300; public Boeing737(int manufactureYear) < this.manufactureYear = manufactureYear; >public int getManufactureYear() < return manufactureYear; >public static class Drawing < private int id; public Drawing(int id) < this.id = id; >public static int getPassengersCount() < return maxPassengersCount; >@Override public String toString() < return "Drawingв документации Oracle. Почитай, если вдруг остались неясные моменты. А теперь самое время решить пару задач! :)