Статические классы
Перевод на русский © , 2000
Java Q&A
Можно ли объявить внутренний или обычный класс как статический (static)?
Для того, чтобы понять использование ключевого слова static в заголовке класса мы должны разобраться с заголовками классов вообще. Вы можете объявить два типа классов: обычные классы (верхнего уровня) и внутренние.
Обычные классы Вы объявляете обычный класс на уровне члена пакета (package). Каждый обычный класс соответствует своему собственному файлу java, использующему тоже имя, что и класс.
Обычный класс по-определению уже находится снаружи, поэтому нет смысла объявлять его статическим; это ошибка. Компилятор определит ее и сообщит вам.
Внутренние классы Вы определяете внутренний класс внутри обычного класса. В зависимости от того, как он определен, внутренний класс может быть одного из четырех типов:
1. Анонимный. Анонимный класс объявляется и создается внутри одного и того же выражения. У них нет имен и они могут быть созданы только один раз.
Вот пример анонимного класса:
okButton.addActionListener( new ActionListener(){ public void actionPerformed (ActionEvent e){ dispose(); } });
Так как анонимный класс не содержит нормального заголовка класса, где можно использовать static, он не может быть объявлен, как статический.
2. Локальный. Локальные классы точно такие-же, как и локальные переменные, в том смысле, что они создаются и используются внутри блока кода. Как только класс внутри блока объявлен, он может быть создан сколько угодно раз внутри этого блока. Как и локальные переменные, локальные классы не могут быть объявлены как public, protected, private или static.
Вот пример кода:
//some code block .......{ class ListListener implements ItemListener { List list; public ListListener(List l) { list = l; }
public void itemStateChanged(ItemEvent e) { String s = l.getItemSelected(); doSomething(s); } } List list1 = new List(); list list2 = new List(); list1.addItemListener( new ListListener(list1)); list2.addItemListener( new ListListener(list2)); }
3. Компонентный. Компонентный класс объявляется внутри тела класса. Вы можете использовать множество классов где-угодно внутри тела класса-контейнера. Вы должны объявить компонентный класс если вы хотите использовать переменные и методы класса-контейнера без необходимости делегирования.
Компонентный класс- единственный, который может быть объявлен статическим. Когда вы объявляете компонентный класс, вы можете создать его только внутри содержимого объекта внешнего класса, по отношению к которому объявлен компонетный класс. Если вы хотите обойти это ограничение, вам необходимо объявить комопонентный класс как статический.
Когда вы объявляете компонентный класс с модификатором static, он становится вложенным классом верхнего уровня и может быть использован как обычный класс, как объяснено выше.
4. Вложенный верхнего уровня. Вложенный класс верхнего уровня это компонентный класс с модификатором static. Вложенный класс верхнего уровня такой-же, как и любой другой класс верхнего уровня, но он объявлен внутри другого класса или интерфейса. Вложенные классы верхнего уровня обычно используются как удобный способ для объединия родственных классов без создания нового пакета.
Если ваш главный класс содержит несколько меньших вспомогательных классов, которые могут быть использованы вне класса и имеют смысл только внутри главного класса, стоит сделать их вложенными классами верхнего уровня. Для использования вложенных классов используйте: КлассВерхнегоУровня.ВложенныйКласс.
Посмотрите на следующий пример:
public class Filter { Vector criteria = new Vector(); public addCriterion(Criterion c) { criteria.addElement(c); } public boolean isTrue(Record rec) { for(Enumeration e=criteria.elements(); e.hasMoreElements();) { if(! ((Criterion)e.nextElement()).isTrue(rec)) return false; } return true; }
public static class Criterion { String colName, colValue; public Criterion(Stirng name, String val) { colName = name; colValue = val; } public boolean isTrue(Record rec) { String data = rec.getData(colName); if(data.equals(colValue)) return true; return false; } } }
И когда вы хотите его использовать:
Filter f = new Filter(); f.addCriterion(new Filter.Criterion("SYMBOL", "SUNW")); f.addCriterion(new Filter.Criterion("SIDE", "BUY")); ..... if(f.isTrue(someRec)) //do some thing .....
Одно важное замечание: ключевое слово static не делает с заголовком класса тоже, что с заголовком пременной или метода.