Java внутренние статические классы

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

Источник

Статические вложенные классы

Java-университет

Статические вложенные классы - 2

Привет! Мы продолжаем изучать тему вложенных классов (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; >> > 

Статические вложенные классы - 4

В этом примере у нас есть внешний класс 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 (велосипед). Без объекта внешнего класса объект внутреннего просто не мог существовать. Для статических вложенных классов это не так. Объект статического вложенного класса вполне может существовать сам по себе. В этом плане статические классы более «независимы», чем нестатические. Единственный момент — при создании такого объекта нужно указывать название внешнего класса:

Читайте также:  Рандомная последовательность чисел python

Почему мы сделали класс 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. Почитай, если вдруг остались неясные моменты. А теперь самое время решить пару задач! :)
John Selawsky
Senior Java-разработчик и преподаватель в LearningTree
Senior Java-разработчик и преподаватель Java на курсах программирования Learning Tree International. Иногда пишу всякие глупости п . [Читать полную биографию]
Комментарии (42)
ЧТОБЫ ПОСМОТРЕТЬ ВСЕ КОММЕНТАРИИ ИЛИ ОСТАВИТЬ КОММЕНТАРИЙ,
ПЕРЕЙДИТЕ В ПОЛНУЮ ВЕРСИЮ
Алексей Голышкин Уровень 33
19 июня 2023
Предыдущая статья про нестатические вложенные классы (1/3 часть): https://javarush.com/groups/posts/2181-vlozhennihe-vnutrennie-klassih Предыдущая статья про нестатические вложенные (анонимные) классы (2/3 часть): https://javarush.com/groups/posts/2193-anonimnihe-klassih
Anonymous #3161756 Уровень 40
6 марта 2023
Судя по комментариям ниже, все валят из авиастроения.
Руслан Шмаков Уровень 108 Expert
4 июля 2022
А на чертеже Ан-124 «Руслан» 😉
Мирослав Уровень 28
9 мая 2022
 public class HandleBar < public void right() < System.out.println("turning to right!"); >public void left() < System.out.println("turning to left!"); >> 
 public class SteeringWheel < public void right() < System.out.println("Руль вправо!"); >public void left() < System.out.println("Руль влево!"); >> 

Источник

Внутренние статические классы

Внутренние статические классы - 1

Перед объявлением внутреннего класса мы можем поставить ключевое слово – static и тогда внутренний класс станет вложенным.

Давай разберемся, что же значит слово static рядом с объявлением вложенного класса. Как ты думаешь?

— Если переменная объявлена статической, то она существует в единственном экземпляре, а если вложенный класс – статический, то можно создать всего один объект такого класса?

— Пусть слово static тут не вводит тебя в заблуждение. Если переменная объявлена статической – то она существуют в единственно экземпляре – это верно. Но статический вложенный класс больше похож на статический метод в этом плане. Слово static перед объявлением класса указывает, что этот класс не хранит в себе ссылок на объекты внешнего класса, внутри которого объявлен.

— Ага. Обычные методы втихаря хранят ссылку на объект, а статические – нет. То же и со статическими классами, я прав, Элли?

— Абсолютно. Твоя догадливость очень похвальна. Вложенные статические классы не имеют скрытых ссылок на объекты внешнего класса, в котором они объявлены.

class Zoo < private static int count = 7; private int mouseCount = 1; public static int getAnimalCount() < return count; >public int getMouseCount() < return mouseCount; >public static class Mouse < public Mouse() < >public int getTotalCount() < return count + mouseCount; //ошибка компиляции. >> >

Давай посмотрим внимательно на этот пример.

К каким переменным может обращаться статический метод getAnimalCount?

— Только к статическим. Это же статический метод.

К каким переменным может обращаться метод getMouseCount?

— И к статическим, и к нестатическим. Он скрытно хранит ссылку(this) на объект типа Zoo.

— Верно. Так вот, статический вложенный класс Mouse, как и статический метод, может обращаться к статическим переменным класса Zoo, но не может обращаться к нестатическим.

Мы можем спокойно создавать объекты класса Mouse, даже когда нет ни одного созданного объекта класса Zoo. Вот как можно это сделать:

Фактически класс Mouse – это самый обычный класс. Из-за того, что он объявлен внутри класса Zoo, у него есть две особенности.

1) При создании объектов вложенного класса (как класс Mouse) вне внешнего класса-родителя, надо еще указывать через точку и имя внешнего класса.

2) Класс Zoo.Mouse и его объекты имеют доступ к private static переменным и методам класса Zoo ( класс Mouse ведь тоже объявлен внутри класса Zoo).

— Т.е. просто дополнительное имя и все?

— Это еще проще, чем казалось на первый взгляд.

Источник

Оцените статью