Definition: Dynamic Binding.
It is not the variable type that determines the method invoked; it is the type of the referenced object. The method to invoke is selected dynamically as the program runs. It is not selected by the compiler.
toggle()
method. The compiler will stop with an error telling us that the Shape does not implement toggle()
. The compiler does not run the program and so it cannot know that our shp
variable actually references a ToggleButton object, so when it is run, the program will not have an error. See the test program in Listing 9.4.1 and the attempt to compile it.// TestToggle.java (version 1)
import doodlepad.*;
public class TestToggle {
public static void main(String[] args) {
// Allowed by dubtype polymorphism
Shape shp = new ToggleButton(100, 100, 100, 50);
// Forbidden by the compiler
shp.toggle();
}
}
TestToggle.java
(version 1)javac -cp doodlepad.jar ToggleButton.java TestToggle.java TestToggle.java:7: error: cannot find symbol shp.toggle(); ^ symbol: method toggle() location: variable shp of type Shape 1 error
shp
back to ToggleButton
before attempting to invoke its toggle()
method. This change of type is accomplished using the cast operator (…)
, in a manner identical to the way we performed widening conversions with primitive types. Naturally, casting a type from a base class to one of its derived classes is legal. See Listing 9.4.2 for the corrected example that compiles and runs without error.// TestToggle.java
import doodlepad.*;
public class TestToggle {
public static void main(String[] args) {
// Allowed by subtype polymorphism
Shape shp = new ToggleButton(100, 100, 100, 50);
// Cast before invoking to make the compiler happy
ToggleButton btn = (ToggleButton) shp;
btn.toggle();
// Forbidden by the compiler
//shp.toggle();
}
}
TestToggle.java
(version 2)onPressed(…)
method and then configured it to handle mouse-pressed events using the setMousePressedHandler(…)
method. This onPressed(…)
method from Listing 14.6.4 is repeated below for convenience.// Handle mouse-pressed event
private void onPressed(Shape shp, double x, double y, int button) {
toggle();
}
name
. The Base class constructor initializes name
to "Base Class" and the Derived class constructor initialize name
to "Derived Class. Both implement a whodunnit()
method. The Derived class version of whodunnit()
overrides the Base class version. Both implementations return name
, which will be the name of the "guilty party." Base implements a main(…)
method that declares a variable named guilty
which has type Base and initializes it to an object of type Derived. It then invokes the whodunnit()
method on the guilty object to find out, whodunnit
.// Base.java
// Demonstration of dynamic binding
// Base class
public class Base {
// Instance variable
private String name;
// Constructor
public Base() {
this.name = "Base Class";
}
// Method
public String whodunnit() {
return this.name;
}
// Run a test
public static void main(String[] args) {
// Declare a Base type variable and
// assign to a Derived type object
Base guilty = new Derived();
// Print the guilty party's name
System.out.println( guilty.whodunnit() );
}
}
// Derived class
class Derived extends Base {
// Instance variable
private String name;
// Constructor
public Derived() {
this.name = "Derived Class";
}
// Overridden Method
public String whodunnit() {
return this.name;
}
}
Base.java
with Derivedjavac Base.java java Base Derived Class
guilty
has type Base, it references an object of type Derived, and so it invokes the Derived class implementation of whodunnit()
. It was not necessary to cast the guilty
variable to Derived
to enure the derived class method was invoked. This happened automatically. This demonstrates dynamic binding in action.guilty
was an instance of Derived
, rather than invoking a method that reveals the secret, we could asked the object directly using an expression like the first in the following code snippet, which prints true. More than the instantiated type, instanceof will also test if an object has a class anywhere up through its inheritance hierarchy. The second expression in the following code snippet also prints true because guilty
inherits Base.System.out.println( guilty instanceof Derived );
System.out.println( guilty instanceof Base );
instanceof
demonstrationsetFontSize(…)
and setFontStyle(…)
on the ToggleButton class even when these methods were not implemented (See Listing 9.3.3). You may not have realized that this is possible only because the base class methods were declared public. Well, this doesn’t seem like much of an advantage, does it? After all, public methods are accessible from anywhere within a program. By comparison, private methods are accessible only from within the implementing class, and not from a derived class. Figure 9.4.5 illustrates this situation.name
. This is unnecessary duplication. In Listing 9.4.7 we change the visibility of the derived class name
instance variable to protected
, which makes the variable visible to the derived class as well. With this enhanced level of visibility, we can now remove the Derived instance variable name
altogether, because Derived can use the one declared in Base. This change in design is more succinct and reflective of the implications of inheritance. If we compile and run this updated program, the outcome is unchanged.// Base.java
// Demonstration of dynamic binding
// Base class
public class Base {
// Instance variable
// private String name;
protected String name; // Updated visibility
// Constructor
public Base() {
this.name = "Base Class";
}
// Method
public String whodunnit() {
return this.name;
}
// Run a test
public static void main(String[] args) {
// Declare a Base type variable and
// assign to a Derived type object
Base guilty = new Derived();
// Print the guilty party's name
System.out.println( guilty.whodunnit() );
}
}
// Derived class
class Derived extends Base {
// Instance variable
// private String name; // Removed
// Constructor
public Derived() {
this.name = "Derived Class";
}
// Overridden Method
public String whodunnit() {
return this.name;
}
}
Base.java
with Derived and protectedjavac Base.java java Base Derived Class
Visibility Modifier | Class/Object | Package | Subclass | World |
---|---|---|---|---|
public |
Yes | Yes | Yes | Yes |
protected |
Yes | Yes | Yes | No |
no modifier | Yes | Yes | No | No |
private |
Yes | No | No | No |