// Поле p получает значение 0.
{ // Блок инициализации экземпляра.
// Выполняется при создании каждого экземпляра // после инициализаций при объявлениях переменных. p = 999; // Этот оператор выполняется в блоке вне всякого метода!
}
static void f(int b){ // Параметр метода b — локальная переменная,
// известная только внутри метода. int a = 2; // Это вторая переменная с тем же именем "a".
// Она известна только внутри метода f()
// и здесь перекрывает первую "a".
int c; // Локальная переменная, известна только в методе f().
// Не получает никакого начального значения // и должна быть определена перед применением.
{ int c = 555; // Сшибка! Попытка повторного объявления.
int x = 333; // Локальная переменная, известна только в этом блоке.
}
// Здесь переменная x уже неизвестна. for (int d = 0; d < 10; d++){
// Переменная цикла d известна только в цикле. int a = 4; // Ошибка!
int e = 5; // Локальная переменная, известна только в цикле for.
e++; // Инициализируется при каждом выполнении цикла.
System.out.println("e = " + e); // Выводится всегда "e = 6".
}
// Здесь переменные d и e неизвестны.
}
public static void main(string[] args){
int a = 9999; // Локальная переменная, известна только внутри
// метода main().
f (a) ;
}
}
Обратите внимание на то, что переменные класса и экземпляра неявно присваивают нулевые значения. Символы неявно получают значение ’ \u0000 ’, логические переменные — значение false, ссылки получают неявно значение null.
Локальные же переменные неявно не инициализируются. Они должны либо явно присваивать значения, либо определяться до первого использования. К счастью, компилятор замечает неопределенные локальные переменные и сообщает о них.
Поля класса при объявлении обнуляются, локальные переменные автоматически не инициализируются.
В листинге 2.6 появилась еще одна новая конструкция:
Вложенные классы
В этой главе уже несколько раз упоминалось, что в теле класса можно сделать описание другого,
□ Можем ли мы из вложенного класса обратиться к членам внешнего класса? Можем, для того это все и задумывалось.
□ А можем ли мы в таком случае определить экземпляр вложенного класса, не определяя экземпляры внешнего класса? Нет, не можем, сначала надо определить хоть один экземпляр внешнего класса, матрешка ведь!
□ А если экземпляров внешнего класса несколько, как узнать, с каким экземпляром внешнего класса работает данный экземпляр вложенного класса? Имя экземпляра вложенного класса уточняется именем связанного с ним экземпляра внешнего класса. Более того, при создании вложенного экземпляра операция new тоже уточняется именем внешнего экземпляра.
□ А?..
Хватит вопросов, давайте разберем все по порядку.
Все вложенные классы можно разделить на вложенные
Классы-члены могут быть объявлены статическими с помощью модификатора static. Поведение статических классов-членов ничем не отличается от поведения обычных классов, отличается только обращение к таким классам. Поэтому они называются
Все нестатические вложенные классы называются
Локальные классы, как и все локальные переменные, известны только в блоке, в котором они определены. Они могут быть
В листинге 2.7 рассмотрены все эти случаи.
class Nested{
static private int pr; // Переменная pr объявлена статической,
// чтобы к ней был доступ из статических классов A и AB.
String s = "Member of Nested";
// Вкладываем статический класс.
static class A{ // Полное имя этого класса — Nested.A
private int a = pr;
String s = "Member of A";