Почему Java не позволяет переопределять статические методы?
Почему невозможно переопределить статические методы?
Если возможно, пожалуйста, используйте пример.
22 ответа
Переопределение зависит от наличия экземпляра класса. Смысл полиморфизма в том, что вы можете создать подкласс класса, и объекты, реализующие эти подклассы, будут иметь разные поведения для одних и тех же методов, определенных в суперклассе (и переопределенных в подклассах). Статический метод не связан ни с одним экземпляром класса, поэтому концепция неприменима.
На разработку Java повлияли два фактора. Одна из них была связана с производительностью: было много критики по поводу того, что Smalltalk слишком медленный (сборщик мусора и полиморфные вызовы являются частью этого), и создатели Java были полны решимости этого избежать. Другим было решение, что целевой аудиторией для Java были разработчики C++. Заставить статические методы работать так, как они это делали, было полезно для программистов на C++, а также было очень быстро, потому что не нужно ждать времени выполнения, чтобы выяснить, какой метод вызывать.
Лично я считаю, что это недостаток в дизайне Java. Да, да, я понимаю, что нестатические методы присоединяются к экземпляру, а статические методы - к классу и т. Д. И т. Д. Тем не менее, рассмотрим следующий код:
public class RegularEmployee {
private BigDecimal salary;
public void setSalary(BigDecimal salary) {
this.salary = salary;
}
public static BigDecimal getBonusMultiplier() {
return new BigDecimal(".02");
}
public BigDecimal calculateBonus() {
return salary.multiply(getBonusMultiplier());
}
/* ... presumably lots of other code ... */
}
public class SpecialEmployee extends RegularEmployee {
public static BigDecimal getBonusMultiplier() {
return new BigDecimal(".03");
}
}
Этот код не будет работать, как вы могли ожидать. А именно, SpecialEmployee получают 2% бонус, как и обычные сотрудники. Но если вы удалите "статические", SpecialEmployee получит бонус 3%.
(Следует признать, что этот пример является плохим стилем кодирования, поскольку в реальной жизни вы, вероятно, захотите, чтобы бонусный множитель находился где-то в базе данных, а не был жестко закодирован. Но это только потому, что я не хотел увязывать пример с большим количеством кода не имеет отношения к делу.)
Мне кажется вполне вероятным, что вы можете сделать getBonusMultiplier статическим. Возможно, вы хотите иметь возможность отображать бонусный множитель для всех категорий сотрудников, без необходимости иметь экземпляр сотрудника в каждой категории. Какой смысл искать такие примеры? Что, если мы создаем новую категорию сотрудников, и у нас еще нет назначенных сотрудников? Это вполне логично, статическая функция.
Но это не работает.
И да, да, я могу придумать множество способов переписать приведенный выше код, чтобы он работал. Моя точка зрения не в том, что это создает неразрешимую проблему, а в том, что она создает ловушку для неосторожного программиста, потому что язык не ведет себя так, как я думаю, что разумный человек ожидал бы.
Возможно, если бы я попытался написать компилятор для языка ООП, я бы быстро понял, почему реализовать его так, чтобы статические функции могли быть переопределены, было бы трудно или невозможно.
Или, возможно, есть какая-то веская причина, почему Java ведет себя так. Кто-нибудь может указать на преимущество в этом поведении, некоторую категорию проблем, которая облегчается этим? Я имею в виду, не просто указывайте мне на спецификацию языка Java и говорите: "Посмотрите, это задокументировано, как оно себя ведет". Я знаю это. Но есть ли веская причина, почему он ДОЛЖЕН вести себя так? (Помимо очевидного "заставить его работать правильно, это было слишком сложно"...)
Обновить
@VicKirk: Если вы имеете в виду, что это "плохой дизайн", потому что он не соответствует тому, как Java обрабатывает статику, мой ответ: "Ну, конечно, конечно". Как я уже говорил в моем оригинальном сообщении, это не работает. Но если вы имеете в виду, что это плохой дизайн в том смысле, что с языком, в котором это работает, было бы что-то принципиально неправильное, то есть где статика могла бы быть переопределена так же, как виртуальные функции, что это каким-то образом внесло бы двусмысленность или было бы невозможно эффективно или что-то подобное, я отвечаю: "Почему? Что не так с концепцией?"
Я думаю, что пример, который я привожу, является очень естественной вещью, которую я хочу сделать. У меня есть класс, у которого есть функция, которая не зависит от каких-либо данных экземпляра, и которую я мог бы очень разумно вызывать независимо от экземпляра, а также желая вызывать из метода экземпляра. Почему это не должно работать? Я сталкивался с этой ситуацией довольно много раз за эти годы. На практике я обхожу это, делая функцию виртуальной, а затем создавая статический метод, единственная цель которого в жизни - быть статическим методом, который передает вызов виртуальному методу с фиктивным экземпляром. Это кажется очень окольным способом добраться туда.
Короткий ответ: это вполне возможно, но Java этого не делает.
Вот некоторый код, который иллюстрирует текущее состояние дел в Java:
файл Base.java
:
package sp.trial;
public class Base {
static void printValue() {
System.out.println(" Called static Base method.");
}
void nonStatPrintValue() {
System.out.println(" Called non-static Base method.");
}
void nonLocalIndirectStatMethod() {
System.out.println(" Non-static calls overridden(?) static:");
System.out.print(" ");
this.printValue();
}
}
файл Child.java
:
package sp.trial;
public class Child extends Base {
static void printValue() {
System.out.println(" Called static Child method.");
}
void nonStatPrintValue() {
System.out.println(" Called non-static Child method.");
}
void localIndirectStatMethod() {
System.out.println(" Non-static calls own static:");
System.out.print(" ");
printValue();
}
public static void main(String[] args) {
System.out.println("Object: static type Base; runtime type Child:");
Base base = new Child();
base.printValue();
base.nonStatPrintValue();
System.out.println("Object: static type Child; runtime type Child:");
Child child = new Child();
child.printValue();
child.nonStatPrintValue();
System.out.println("Class: Child static call:");
Child.printValue();
System.out.println("Class: Base static call:");
Base.printValue();
System.out.println("Object: static/runtime type Child -- call static from non-static method of Child:");
child.localIndirectStatMethod();
System.out.println("Object: static/runtime type Child -- call static from non-static method of Base:");
child.nonLocalIndirectStatMethod();
}
}
Если вы запустите это (я сделал это на Mac, из Eclipse, используя Java 1.6), вы получите:
Object: static type Base; runtime type Child.
Called static Base method.
Called non-static Child method.
Object: static type Child; runtime type Child.
Called static Child method.
Called non-static Child method.
Class: Child static call.
Called static Child method.
Class: Base static call.
Called static Base method.
Object: static/runtime type Child -- call static from non-static method of Child.
Non-static calls own static.
Called static Child method.
Object: static/runtime type Child -- call static from non-static method of Base.
Non-static calls overridden(?) static.
Called static Base method.
Здесь единственными случаями, которые могут быть неожиданностью (и о чем идет речь), является первый случай:
"Тип времени выполнения не используется для определения, какие статические методы вызываются, даже если они вызваны с экземпляром объекта (obj.staticMethod()
) ".
и последний случай:
"При вызове статического метода из метода объекта класса выбран статический метод, доступный из самого класса, а не из класса, определяющего тип объекта во время выполнения".
Вызов с экземпляром объекта
Статический вызов разрешается во время компиляции, тогда как вызов нестатического метода разрешается во время выполнения. Обратите внимание, что хотя статические методы наследуются (от родителя), они не переопределяются (дочерними). Это может быть сюрпризом, если вы ожидаете иначе.
Вызов из метода объекта
Вызовы методовобъекта разрешаются с использованием типа времени выполнения, но вызовы статических (классовых) методов разрешаются с использованием типа (объявленного) времени компиляции.
Изменение правил
Чтобы изменить эти правила, чтобы последний вызов в примере Child.printValue()
статические вызовы должны быть предоставлены с типом во время выполнения, а не компилятором, разрешающим вызов во время компиляции с объявленным классом объекта (или контекста). Статические вызовы могут затем использовать (динамическую) иерархию типов для разрешения вызова, как это делают вызовы объектных методов сегодня.
Это было бы легко выполнимо (если мы изменили Java:-O), и это вовсе не неразумно, однако, у него есть некоторые интересные соображения.
Основное соображение заключается в том, что нам нужно решить, какие статические вызовы методов должны делать это.
На данный момент у Java есть такая "причуда" на языке, посредством которого obj.staticMethod()
звонки заменяются ObjectClass.staticMethod()
звонки (обычно с предупреждением). [Примечание: ObjectClass
тип времени компиляции obj
.] Это были бы хорошие кандидаты для переопределения таким образом, принимая тип времени выполнения obj
,
Если бы мы сделали это, это бы затруднило чтение тел методов: статические вызовы в родительском классе потенциально могли бы динамически "перенаправляться". Чтобы избежать этого, нам пришлось бы вызывать статический метод с именем класса - и это делает вызовы более очевидно разрешенными с помощью иерархии типов времени компиляции (как сейчас).
Другие способы вызова статического метода более сложны: this.staticMethod()
должно означать так же, как obj.staticMethod()
, принимая тип времени выполнения this
, Однако это может вызвать некоторые проблемы с существующими программами, которые вызывают (по-видимому, локальные) статические методы без декорации (что, возможно, эквивалентно this.method()
).
Так что насчет неукрашенных звонков staticMethod()
? Я предлагаю, чтобы они делали то же, что и сегодня, и использовали контекст локального класса, чтобы решить, что делать. В противном случае возникнет великое замешательство. Конечно, это означает, что method()
будет означать this.method()
если method
был нестатический метод, и ThisClass.method()
если method
были статическим методом. Это еще один источник путаницы.
Другие соображения
Если бы мы изменили это поведение (и сделали статические вызовы потенциально динамически нелокальными), мы бы, вероятно, захотели бы вернуться к значению final
, private
а также protected
в качестве квалификаторов на static
методы класса. Тогда нам всем придется привыкнуть к тому, что private static
а также public final
методы не переопределяются и поэтому могут быть безопасно разрешены во время компиляции, и их "безопасно" читать как локальные ссылки.
На самом деле мы были не правы.
Несмотря на то, что Java не позволяет вам переопределять статические методы по умолчанию, если вы внимательно изучите документацию классов Class и Method в Java, вы все равно сможете найти способ эмулировать переопределение статических методов, выполнив следующий обходной путь:
import java.lang.reflect.InvocationTargetException;
import java.math.BigDecimal;
class RegularEmployee {
private BigDecimal salary = BigDecimal.ONE;
public void setSalary(BigDecimal salary) {
this.salary = salary;
}
public static BigDecimal getBonusMultiplier() {
return new BigDecimal(".02");
}
public BigDecimal calculateBonus() {
return salary.multiply(this.getBonusMultiplier());
}
public BigDecimal calculateOverridenBonus() {
try {
// System.out.println(this.getClass().getDeclaredMethod(
// "getBonusMultiplier").toString());
try {
return salary.multiply((BigDecimal) this.getClass()
.getDeclaredMethod("getBonusMultiplier").invoke(this));
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
}
return null;
}
// ... presumably lots of other code ...
}
final class SpecialEmployee extends RegularEmployee {
public static BigDecimal getBonusMultiplier() {
return new BigDecimal(".03");
}
}
public class StaticTestCoolMain {
static public void main(String[] args) {
RegularEmployee Alan = new RegularEmployee();
System.out.println(Alan.calculateBonus());
System.out.println(Alan.calculateOverridenBonus());
SpecialEmployee Bob = new SpecialEmployee();
System.out.println(Bob.calculateBonus());
System.out.println(Bob.calculateOverridenBonus());
}
}
Результирующий вывод:
0.02
0.02
0.02
0.03
что мы пытались достичь:)
Даже если мы объявим третью переменную Carl как RegularEmployee и назначим ей экземпляр SpecialEmployee, у нас все равно будет вызов метода RegularEmployee в первом случае и вызов метода SpecialEmployee во втором случае
RegularEmployee Carl = new SpecialEmployee();
System.out.println(Carl.calculateBonus());
System.out.println(Carl.calculateOverridenBonus());
просто посмотрите на консоль вывода:
0.02
0.03
;)
Статические методы обрабатываются JVM как глобальные, они вообще не привязаны к экземпляру объекта.
Концептуально это было бы возможно, если бы вы могли вызывать статические методы из объектов класса (как в языках, таких как Smalltalk), но это не так в Java.
РЕДАКТИРОВАТЬ
Вы можете перегрузить статический метод, это нормально. Но вы не можете переопределить статический метод, потому что класс не первоклассный объект. Вы можете использовать отражение, чтобы получить класс объекта во время выполнения, но полученный вами объект не параллелен иерархии классов.
class MyClass { ... }
class MySubClass extends MyClass { ... }
MyClass obj1 = new MyClass();
MySubClass obj2 = new MySubClass();
ob2 instanceof MyClass --> true
Class clazz1 = obj1.getClass();
Class clazz2 = obj2.getClass();
clazz2 instanceof clazz1 --> false
Вы можете размышлять над классами, но это останавливается там. Вы не вызываете статический метод, используя clazz1.staticMethod()
, но используя MyClass.staticMethod()
, Статический метод не привязан к объекту и, следовательно, нет понятия this
ни super
в статическом методе. Статический метод - это глобальная функция; как следствие, также отсутствует понятие полиморфизма и, следовательно, переопределение метода не имеет смысла.
Но это может быть возможно, если MyClass
был объектом во время выполнения, для которого вы вызываете метод, как в Smalltalk (или, может быть, JRuby, как предполагает один комментарий, но я ничего не знаю о JRuby).
Ах да... еще одна вещь. Вы можете вызвать статический метод через объект obj1.staticMethod()
но это действительно синтаксический сахар для MyClass.staticMethod()
и следует избегать. Это обычно вызывает предупреждение в современной IDE. Я не знаю, почему они позволили этот ярлык.
Переопределение метода стало возможным благодаря динамической диспетчеризации, что означает, что объявленный тип объекта не определяет его поведение, а скорее тип времени выполнения:
Animal lassie = new Dog();
lassie.speak(); // outputs "woof!"
Animal kermit = new Frog();
kermit.speak(); // outputs "ribbit!"
Хотя оба lassie
а также kermit
объявлены как объекты типа Animal
, их поведение (метод .speak()
) меняется, потому что динамическая диспетчеризация будет связывать только вызов метода .speak()
реализации во время выполнения - не во время компиляции.
Теперь вот где static
Ключевое слово начинает иметь смысл: слово "статический" является антонимом для "динамический". Поэтому причина, по которой вы не можете переопределить статические методы, заключается в том, что нет динамической диспетчеризации статических элементов - потому что статический буквально означает "не динамический". Если они отправлены динамически (и, таким образом, могут быть переопределены), static
Ключевое слово просто не имеет смысла больше.
Да. Практически Java допускает переопределение статического метода, и теоретически Нет, если вы переопределяете статический метод в Java, тогда он будет скомпилирован и работать без сбоев, но потеряет полиморфизм, который является базовым свойством Java. Вы будете читать везде, что вы не можете попробовать самостоятельно скомпилировать и запустить. вы получите ответ. Например, если у вас есть Class Animal и статический метод eat(), и вы переопределяете этот статический метод в его подклассе, то вы называете его Dog. Тогда, когда бы вы ни назначали объект Dog для ссылки на животных и вызывали eat() в соответствии с Java, должна быть вызвана функция eat() собаки, но в статических переопределенных животных будет вызываться функция eat().
class Animal {
public static void eat() {
System.out.println("Animal Eating");
}
}
class Dog extends Animal{
public static void eat() {
System.out.println("Dog Eating");
}
}
class Test {
public static void main(String args[]) {
Animal obj= new Dog();//Dog object in animal
obj.eat(); //should call dog's eat but it didn't
}
}
Output Animal Eating
Согласно принципу полиморфизма Java, результат должен быть Dog Eating
,
Но результат был другим, потому что для поддержки полиморфизма Java использует Late Binding, что означает, что методы вызываются только во время выполнения, но не в случае статических методов. В статических методах компилятор вызывает методы во время компиляции, а не во время выполнения, поэтому мы получаем методы в соответствии со ссылкой, а не со ссылкой на объект, содержащий ссылку, поэтому вы можете сказать, что практически он поддерживает статическое переопределение, но теоретически это не так. "т.
В Java (и во многих языках ООП, но я не могу говорить за всех; а некоторые вообще не имеют статики) все методы имеют фиксированную сигнатуру - параметры и типы. В виртуальном методе подразумевается первый параметр: ссылка на сам объект и при вызове изнутри объекта компилятор автоматически добавляет this
,
Для статических методов нет никакой разницы - они все еще имеют фиксированную подпись. Однако, объявив метод статическим, вы явно заявили, что компилятор не должен включать параметр подразумеваемого объекта в начале этой сигнатуры. Поэтому любой другой код, который вызывает это, не должен пытаться поместить ссылку на объект в стек. Если бы он это сделал, то выполнение метода не сработало бы, поскольку параметры были бы в неправильном месте - смещены на единицу - в стеке.
Из-за этой разницы между ними; виртуальные методы всегда имеют ссылку на объект контекста (т.е. this
), тогда в кучи можно ссылаться на все, что принадлежит этому экземпляру объекта. Но со статическими методами, поскольку не передается ссылка, этот метод не может получить доступ к переменным и методам объекта, так как контекст неизвестен.
Если вы хотите, чтобы Java изменила определение так, чтобы контекст объекта передавался для каждого метода, статического или виртуального, то в сущности вы бы имели только виртуальные методы.
Как кто-то спросил в комментарии к оперу - какова ваша причина и цель, чтобы хотеть эту функцию?
Я не знаю много Ruby, так как это было упомянуто OP, я провел некоторое исследование. Я вижу, что в классах Ruby действительно особый вид объектов, и можно создавать (даже динамически) новые методы. Классы являются полными объектами классов в Ruby, их нет в Java. Это то, что вы должны будете принять при работе с Java (или C#). Это не динамические языки, хотя C# добавляет некоторые формы динамических. На самом деле, в Ruby нет "статических" методов, насколько я мог найти - в этом случае это методы объекта-синглтона. Затем вы можете переопределить этот синглтон новым классом, и методы в предыдущем объекте класса будут вызывать методы, определенные в новом классе (правильно?). Поэтому, если вы вызываете метод в контексте исходного класса, он все равно будет выполнять только исходную статику, но вызов метода в производном классе вызовет методы либо из родительского, либо из подкласса. Интересно, и я вижу в этом некоторую ценность. Требуется другая модель мышления.
Поскольку вы работаете в Java, вам нужно будет приспособиться к такому способу работы. Почему они это сделали? Ну, возможно, чтобы улучшить производительность в то время на основе технологии и понимания, которые были доступны. Компьютерные языки постоянно развиваются. Вернитесь достаточно далеко, и ООП не будет. В будущем появятся другие новые идеи.
РЕДАКТИРОВАТЬ: еще один комментарий. Теперь, когда я вижу различия и, будучи самим разработчиком Java/C#, я могу понять, почему ответы, которые вы получаете от разработчиков Java, могут сбивать с толку, если вы работаете с таким языком, как Ruby. Джава static
методы не такие как Ruby class
методы. Разработчикам Java будет трудно понять это, так же как и тем, кто работает в основном с таким языком, как Ruby/Smalltalk. Я могу видеть, как это могло бы сильно сбить с толку тем фактом, что Java также использует "метод класса" как еще один способ говорить о статических методах, но этот же термин по-разному используется в Ruby. В Java нет методов класса Ruby (извините); Ruby не имеет статических методов в стиле Java, которые на самом деле являются просто старыми функциями процедурного стиля, как в C.
Кстати - спасибо за вопрос! Сегодня я узнал что-то новое о методах класса (стиль Ruby).
Переопределение зарезервировано для членов экземпляра для поддержки полиморфного поведения. статические члены класса не принадлежат конкретному экземпляру. вместо этого статические члены принадлежат классу, и в результате переопределение не поддерживается, поскольку подклассы наследуют только защищенные и открытые члены экземпляра, а не статические члены. Вы можете захотеть определить внутреннюю и исследовательскую фабрику и / или шаблоны разработки стратегии для оценки альтернативного подхода.
Мне нравится и двойной комментарий Джея ( /questions/40795107/pochemu-java-ne-pozvolyaet-pereopredelyat-staticheskie-metodyi/40795122#40795122).
Я согласен, что это плохой дизайн Java.
Многие другие языки поддерживают переопределение статических методов, как мы видели в предыдущих комментариях. Я чувствую, что Джей также пришел на Яву из Delphi, как и я.
Delphi (Object Pascal) был первым языком, реализующим ООП.
Очевидно, что многие люди имели опыт работы с этим языком, поскольку в прошлом он был единственным языком для написания коммерческих продуктов с графическим интерфейсом. И - да, мы могли бы в Delphi переопределить статические методы. На самом деле статические методы в Delphi называются "методами класса", в то время как Delphi придерживалась другого понятия "статические методы Delphi", которые были методами с ранним связыванием. Чтобы переопределить методы, которые вам пришлось использовать с поздним связыванием, объявите "виртуальную" директиву. Так что это было очень удобно и интуитивно понятно, и я ожидал этого на Java.
Ну... ответ НЕТ, если вы думаете с точки зрения того, как переопределенный метод должен вести себя в Java. Но вы не получите никакой ошибки компилятора, если попытаетесь переопределить статический метод. Это означает, что если вы попытаетесь переопределить, Java не остановит вас; но вы, конечно, не получите тот же эффект, что и для нестатических методов. Переопределение в Java просто означает, что конкретный метод будет вызываться на основе типа времени выполнения объекта, а не типа его времени компиляции (что имеет место с переопределенными статическими методами). Хорошо... есть предположения, почему они ведут себя странно? Поскольку они являются методами класса и, следовательно, доступ к ним всегда разрешается во время компиляции только с использованием информации о типе времени компиляции. Доступ к ним с использованием ссылок на объекты - это просто дополнительная свобода, предоставляемая разработчиками Java, и мы, конечно, не должны думать о том, чтобы прекращать эту практику только тогда, когда они ограничивают ее:-)
Пример: давайте попробуем посмотреть, что произойдет, если мы попробуем переопределить статический метод:
class SuperClass {
// ......
public static void staticMethod() {
System.out.println("SuperClass: inside staticMethod");
}
// ......
}
public class SubClass extends SuperClass {
// ......
// overriding the static method
public static void staticMethod() {
System.out.println("SubClass: inside staticMethod");
}
// ......
public static void main(String[] args) {
// ......
SuperClass superClassWithSuperCons = new SuperClass();
SuperClass superClassWithSubCons = new SubClass();
SubClass subClassWithSubCons = new SubClass();
superClassWithSuperCons.staticMethod();
superClassWithSubCons.staticMethod();
subClassWithSubCons.staticMethod();
// ...
}
}
Выход: -SuperClass: inside staticMethod
SuperClass: inside staticMethod
SubClass: inside staticMethod
Обратите внимание на вторую строку вывода. Если бы staticMethod был переопределен, эта строка должна быть идентична третьей строке, так как мы вызываем функцию staticMethod() для объекта Runtime Type как "SubClass", а не как "SuperClass". Это подтверждает, что статические методы всегда разрешаются с использованием только их информации о типе времени компиляции.
В общем случае не имеет смысла разрешать "переопределение" статических методов, поскольку не было бы хорошего способа определить, какой из них вызывать во время выполнения. Если взять пример Employee, если мы вызовем RegularEmployee.getBonusMultiplier() - какой метод должен быть выполнен?
В случае Java можно представить определение языка, в котором можно "переопределить" статические методы, если они вызываются через экземпляр объекта. Однако все, что нужно сделать, - это повторно реализовать обычные методы класса, добавив избыточность к языку, не добавляя никакой выгоды.
Переопределением мы можем создать полиморфную природу в зависимости от типа объекта. Статический метод не имеет отношения к объекту. Так что Java не может поддерживать переопределение статического метода.
Переопределив, вы достигните динамического полиморфизма. Когда вы говорите о переопределении статических методов, слова, которые вы пытаетесь использовать, противоречивы.
Статика говорит - время компиляции, переопределение используется для динамического полиморфизма. Оба являются противоположными по природе, и, следовательно, не могут быть использованы вместе.
Динамическое полиморфное поведение возникает, когда программист использует объект и обращается к методу экземпляра. JRE отобразит разные методы экземпляров разных классов в зависимости от того, какой тип объекта вы используете.
Когда вы говорите о переопределении статических методов, мы обращаемся к статическим методам, используя имя класса, которое будет связано во время компиляции, поэтому нет понятия связывания методов во время выполнения со статическими методами. Таким образом, сам термин "переопределение" статических методов не имеет никакого значения.
Примечание: даже если вы обращаетесь к методу класса с помощью объекта, все же Java-компилятор достаточно умен, чтобы это выяснить, и будет выполнять статическое связывание.
Что хорошего в этом, чтобы переопределить статические методы. Вы не можете вызывать статические методы через экземпляр.
MyClass.static1()
MySubClass.static1() // If you overrode, you have to call it through MySubClass anyway.
РЕДАКТИРОВАТЬ: Кажется, что из-за неудачного надзора за языковой дизайн, вы можете вызывать статические методы через экземпляр Обычно никто не делает это. Виноват.
Переопределение в Java просто означает, что конкретный метод будет вызываться в зависимости от типа объекта во время выполнения, а не от его типа во время компиляции (что имеет место в случае переопределенных статических методов). Так как статические методы являются методами класса, они не являются методами экземпляра, поэтому они не имеют никакого отношения к тому факту, какая ссылка указывает на какой объект или экземпляр, потому что из-за природы статического метода он принадлежит определенному классу. Вы можете переопределить его в подклассе, но этот подкласс не будет ничего знать о статических методах родительского класса, потому что, как я сказал, он специфичен только для того класса, в котором он был объявлен. Доступ к ним с использованием ссылок на объекты - это просто дополнительная свобода, предоставляемая разработчиками Java, и мы, конечно, не должны отказываться от этой практики только тогда, когда они ограничивают ее более подробной информацией и примером http://faisalbhagat.blogspot.com/2014/09/method-overriding-and-method-hiding.html
Ответ на этот вопрос прост: метод или переменная, помеченная как статическая, принадлежит только классу, поэтому статический метод нельзя наследовать в подклассе, поскольку он принадлежит только суперклассу.
Простое решение: использовать экземпляр Singleton. Это позволит переопределения и наследования.
В моей системе у меня есть класс SingletonsRegistry, который возвращает экземпляр для переданного класса. Если экземпляр не найден, он создается.
Haxe языковой класс:
package rflib.common.utils;
import haxe.ds.ObjectMap;
class SingletonsRegistry
{
public static var instances:Map<Class<Dynamic>, Dynamic>;
static function __init__()
{
StaticsInitializer.addCallback(SingletonsRegistry, function()
{
instances = null;
});
}
public static function getInstance(cls:Class<Dynamic>, ?args:Array<Dynamic>)
{
if (instances == null) {
instances = untyped new ObjectMap<Dynamic, Dynamic>();
}
if (!instances.exists(cls))
{
if (args == null) args = [];
instances.set(cls, Type.createInstance(cls, args));
}
return instances.get(cls);
}
public static function validate(inst:Dynamic, cls:Class<Dynamic>)
{
if (instances == null) return;
var inst2 = instances[cls];
if (inst2 != null && inst != inst2) throw "Can\'t create multiple instances of " + Type.getClassName(cls) + " - it's singleton!";
}
}
Вот простое объяснение. Статический метод связан с классом, а метод экземпляра связан с конкретным объектом. Переопределения позволяют вызывать различные реализации переопределенных методов, связанных с конкретным объектом. Так что нелогично переопределять статический метод, который даже не связан с объектами, но сам класс в первую очередь. Поэтому статические методы не могут быть переопределены в зависимости от того, какой объект их вызывает, они всегда будут связаны с классом, в котором они были созданы.
Статический метод, переменная, блок или вложенный класс принадлежит всему классу, а не объекту.
Метод в Java используется для представления поведения объекта / класса. Здесь, поскольку метод является статическим (т. Е. Статический метод используется для представления поведения только класса.) Изменение / переопределение поведения всего класса нарушит явление одного из фундаментальных элементов объектно-ориентированного программирования, то есть высокой сплоченности, (помните, конструктор - это особый метод в Java.)
Высокая сплоченность - один класс должен иметь только одну роль. Например: автомобильный класс должен производить только автомобильные объекты, а не велосипед, грузовики, самолеты и т. Д. Но у автомобильного класса могут быть некоторые особенности (поведение), которые принадлежат только ему.
Поэтому пока проектирую язык программирования java. Разработчики языка решили позволить разработчикам сохранить некоторые поведения класса при себе, только сделав метод статичным по своей природе.
Приведенный ниже фрагмент кода пытается переопределить статический метод, но не встретит никакой ошибки компиляции.
public class Vehicle {
static int VIN;
public static int getVehileNumber() {
return VIN;
}}
class Car extends Vehicle {
static int carNumber;
public static int getVehileNumber() {
return carNumber;
}}
Это потому, что здесь мы не переопределяем метод, а просто объявляем его заново. Java позволяет повторно объявить метод (статический / нестатический).
Удаление статического ключевого слова из метода getVehileNumber() класса Car приведет к ошибке компиляции, поскольку мы пытаемся изменить функциональность статического метода, который принадлежит только классу Vehicle.
Кроме того, если getVehileNumber() объявлен как final, код не будет компилироваться, поскольку ключевое слово final ограничивает программиста от повторного объявления метода.
public static final int getVehileNumber() {
return VIN; }
В целом, это зависит от разработчиков программного обеспечения, где использовать статические методы. Я лично предпочитаю использовать статические методы для выполнения некоторых действий без создания какого-либо экземпляра класса. Во-вторых, чтобы скрыть поведение класса от внешнего мира.
Теперь, увидев ответы выше, все знают, что мы не можем переопределить статические методы, но не следует неправильно понимать концепцию доступа к статическим методам из подкласса.
Мы можем получить доступ к статическим методам суперкласса со ссылкой на подкласс, если этот статический метод не был скрыт новым статическим методом, определенным в подклассе.
Например, см. Ниже код:-
public class StaticMethodsHiding {
public static void main(String[] args) {
SubClass.hello();
}
}
class SuperClass {
static void hello(){
System.out.println("SuperClass saying Hello");
}
}
class SubClass extends SuperClass {
// static void hello() {
// System.out.println("SubClass Hello");
// }
}
Выход:-
SuperClass saying Hello
См. Документацию Java oracle и поиск " Что можно сделать в подклассе" для получения подробной информации о сокрытии статических методов в подклассе.
Спасибо
Следующий код показывает, что это возможно:
class OverridenStaticMeth {
static void printValue() {
System.out.println("Overriden Meth");
}
}
public class OverrideStaticMeth extends OverridenStaticMeth {
static void printValue() {
System.out.println("Overriding Meth");
}
public static void main(String[] args) {
OverridenStaticMeth osm = new OverrideStaticMeth();
osm.printValue();
System.out.println("now, from main");
printValue();
}
}