Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to call abstract method from abstract class called by inherit class

I want to call a method of an abstract class from abstract class called by inherit class.

Abstract class:

public abstract class Abstract {
    protected void updateMotionY(float deltaTime) {
        System.out.println("Abstrcat updateMotionY");
    }
    public void update(float deltaTime) {
        this.updateMotionY(deltaTime);
    }
}

Inherit class:

public class Obj extends Abstract {
    @Override
    protected void updateMotionY(float deltaTime) {
        System.out.println("updateMotionY");
        super.updateMotionY(deltaTime);
    }
    @Override
    public void update(float deltaTime) {
        super.update(deltaTime);
    }
}

Main method class:

public static void main(String[] args) {
    (new Obj()).update(10.0f);
}

Whenever I try to call new Obj().update() method in main class, it prints "updateMotionY" and "Abstrcat updateMotionY". I want to get only "Abstrcat updateMotionY".

Can anyone tell me how to resolve this problem?

like image 276
Amadeus Avatar asked Aug 16 '17 11:08

Amadeus


2 Answers

I think you are using abstract in a very wrong way. Your base class should rather look like this:

public abstract class Abstract {
  protected abstract void updateMotionY(float deltaTime);

  public final void update(float deltaTime) {
    this.updateMotionY(deltaTime);
  }
}

Notes:

  • there is no point putting print "is abstract" into an abstract method. The java language has a keyword to express this fact.
  • subclasses should only be about implementing the abstract method(s) (probably in different ways). You absolutely do not want that subclasses change the implementation of other methods of the base class. Your base class defines a contract - and subclasses should adhere to that (as outlined by the Liskov Substitution Principle).

In other words: put the common parts solely in the base class, and make sure that you have to necessary abstract methods in there to do that. But avoid implementing methods more than once. That only leads to confusion and strange bugs.

like image 113
GhostCat Avatar answered Oct 14 '22 09:10

GhostCat


(new Obj()).update(10.0f) calls Obj::update which calls Abstract::update which calls this.updateMotionY. Because this is an instance of Obj, this calls Obj::updateMotionY.

This prints "updateMotionY".

This then calls super.updateMotionY(deltaTime) which is Abstract::updateMotionY.

This prints "Abstrcat updateMotionY".

That's the end of the call hierarchy and everything unwinds.


Fundamentally your confusion seems to stem from the fact that this.updateMotionY(deltaTime); in the Abstract class resolves to updateMotionY in the Obj class. That's basically the whole point of polymorphism.


One thing you could do is to add a private method (so that it cant be overridden) which contains the actual implementation, and defer to it:

public abstract class Abstract {
    private void motionY(float dt)
    {
        System.out.println("Abstrcat updateMotionY");
    }

    protected void updateMotionY(float deltaTime) {
        motionY(deltaTime);
    }
    public void update(float deltaTime) {
        motionY(deltaTime);
    }
}
like image 38
Michael Avatar answered Oct 14 '22 07:10

Michael