Я пытаюсь понять, как Java выбирает, какой метод выполняется:
//Example 1 prints Square:add(Figure)
Figure fs = new Square();
fs.add(fs);
//Example 2 prints Square:add(Figure)
Rectangle rs = new Square();
rs.add(fs);
//Example 3 prints Rectangle:add(Rectangle). Expected Square:add(Square)
rs.add(new Square());
//Example 4 prints Rectangle:add(Rectangle). Expected Square:add(Figure)
Square ss = new Square();
ss.add(rs);
class Figure
{
public void add(Figure f){ System.out.println("Figure:add(Figure)"); }
}
class Rectangle extends Figure
{
@Override
public void add(Figure f){ System.out.println("Rectangle:add(Figure)"); }
public void add(Rectangle r){ System.out.println("Rectangle:add(Rectangle)"); }
}
class Square extends Rectangle
{
@Override
public void add(Figure f){ System.out.println("Square:add(Figure)"); }
public void add(Square s){ System.out.println("Square:add(Square)"); }
}
То, что я узнал здесь,
- Подпись метода определяется по типам данных времени компиляции
- Вызываемый действительный метод зависит от динамического типа объекта, на который вызывается метод.
Исходя из этого, результат первых двух вызовов соответствует ожиданиям. Однако я не понимаю результат из примеров 3 и 4.
Как представляется, в спецификации java language, но я этого не понимаю.