In this tutorial we will discuss about the inheritance in Java. The most fundamental element of Java is the class. A class represents an entity and also, defines and implements its functionality. In Java, classes can be derived from other classes, in order to create more complex relationships.
A class that is derived from another class is called subclass and inherits all fields and methods of its superclass. In Java, only single inheritance is allowed and thus, every class can have at most one direct superclass. A class can be derived from another class that is derived from another class and so on. Finally, we must mention that each class in Java is implicitly a subclass of the Object
class.
Suppose we have declared and implemented a class A. In order to declare a class B that is derived from A, Java offers the extendkeyword that is used as shown below:
Java supports only public inheritance and thus, all fields and methods of the superclass are inherited and can be used by the subclass. The only exception are the private members of the superclass that cannot be accessed directly from the subclass. Also, constructors are not members, thus they are not inherited by subclasses, but the constructor of the superclass can be invoked from the subclass. In order to call the constructor of the superclass Java provides the keyword super, as shown below:
03 |
System.out.println( "New A" ); |
10 |
System.out.println( "New B" ); |
A sample example to present the inheritance in Java is shown below:
Animal.java:
03 |
System.out.println( "A new animal has been created!" ); |
07 |
System.out.println( "An animal sleeps..." ); |
11 |
System.out.println( "An animal eats..." ); |
Bird.java:
01 |
public class Bird extends Animal { |
04 |
System.out.println( "A new bird has been created!" ); |
09 |
System.out.println( "A bird sleeps..." ); |
14 |
System.out.println( "A bird eats..." ); |
Dog.java:
01 |
public class Dog extends Animal { |
04 |
System.out.println( "A new dog has been created!" ); |
09 |
System.out.println( "A dog sleeps..." ); |
14 |
System.out.println( "A dog eats..." ); |
MainClass.java:
01 |
public class MainClass { |
02 |
public static void main(String[] args) { |
03 |
Animal animal = new Animal(); |
04 |
Bird bird = new Bird(); |
In this example we created three distinct classes, Animal
, Dog
and Bird
. Both Dog
and Bird
classes extend the Animal
class and thus, they inherit its members and methods. Moreover, as we can see below, each class overrides the methods of Animal
and thus, both the Dog and Bird classes redefine the functionality of Animal’s
methods.
A sample execution is shown below:
A new animal has been created!
A new animal has been created!
A new bird has been created!
A new animal has been created!
A new dog has been created!
An animal sleeps...
An animal eats...
A bird sleeps...
A bird eats...
A dog sleeps...
A dog eats...
A nested class has access to all the private members of its enclosing class, both fields and methods. Therefore, a public or protected nested class inherited by a subclass has indirect access to all of the private members of the superclass.
As already mentioned, a subclass inherits all of the public and protected members of its superclass. If the subclass is in the same package as its superclass, it also inherits the package-private members of the superclass. The inheritance in Java provides the following features:
- You can declare a field in the subclass with the same name as the one in the superclass thus, hiding it. This is called shadowing.
- You can declare new fields in the subclass that are not in the superclass.
- You can write a new instance method in the subclass that has the same signature as the one in the superclass, thus overriding it.
- You can declare new methods in the subclass that are not in the superclass.
Final abstract classes can exist in a hierarchy of types. For more information about abstract classes and how are used in Java, please refer to the Java abstract tutorial here.
Inheritance and Casting
When a class B extends a class A, then an instance of the B class is of type B, but also of type A. Thus, such an instance can be used in all cases where a class B or class A object is required. However, the reverse is not true! An instance of the class A is of course of type A, but it is not of type B.
Thus, we can use casting between the instances of classes. The cast inserts a runtime check, in order for the compiler to safely assume that the cast is used properly and is correct. If not, a runtime exception will be thrown.
A simple example that demonstrates the usage of casting is shown below:
2 |
Animal a2 = new Bird(); |
A sample execution is shown below:
A dog eats...
A bird sleeps...
The instanceof operator
The instanceof operator can be used, in order to determine if an object is a valid instance of a specific type. It can be used to test if an object is an instance of a class, an instance of a subclass, or an instance of a class that implements a particular interface. A simple example is shown below:
2 |
if (d instanceof Animal) { |
Interfaces
An interface in Java is an abstract type that is used to declare and specify a set of public methods and members. An interface can be implemented by a class. In this case, the class must provide an implementation for every method defined in the interface. A significant advantage of using interfaces is the fact that in Java, multiple interfaces can be implemented by a single class.
A sample example that uses both classes and multiple interfaces is shown below:
BasketballTeam.java:
1 |
public interface BasketballTeam { |
2 |
public void printBasketballName(); |
FootballTeam.java:
1 |
public interface FootballTeam { |
2 |
public void printFootballName(); |
Team.java:
01 |
public class Team implements BasketballTeam, FootballTeam { |
03 |
private String name = null ; |
05 |
public Team(String name) { |
10 |
public void printFootballName() { |
11 |
System.out.println( "Football Team: \"" + name + " F.C.\"" ); |
15 |
public void printBasketballName() { |
16 |
System.out.println( "Basketball Team: \"" + name + " B.C.\"" ); |
19 |
public static void main(String[] args) { |
20 |
Team t = new Team( "Team A" ); |
21 |
t.printBasketballName(); |
22 |
t.printFootballName(); |
A sample execution is shown below:
Basketball Team: "Team A B.C."
Football Team: "Team A F.C."