Java Abstract Class Example (original) (raw)

In this tutorial, we will discuss abstraction in java through examples. We are also going to talk about abstract class vs interface implementation in Java.

Data abstraction is the process of hiding certain details and showing only essential information to the user. We can have abstraction with using either abstract classes or interfaces.

An abstract class is a class that is declared using the abstract keyword and it cannot be instantiated. It can be used only as a super-class for those classes that extend the abstract class. The default functionality of the class still exists, with its fields, methods and constructors being accessed in the same way as with the other classes.

Moreover, an abstract class may contain methods without any implementation, called abstract methods. The declaration of an abstract method starts with the abstract keyword and ends with a semicolon, instead of the method’s body. If a class contains an abstract method, either declared or inherited, it must be declared as an abstract class.

A class that extends an abstract class must implement all its abstract methods (if any). Otherwise, the sub-class must be declared as abstract as well. Finally, any implementation of an abstract method can be overridden by additional sub-classes.

You can also check this tutorial in the following video:

Java Abstract Class Example – Video

The last things to mention are that abstract classes can also implement methods, despite providing just their signature and that an abstract class may have static fields and static methods.

In this example, we use the following tools on a Windows 7 platform:

1. Using a Java Abstract class

The purpose of an abstract class is to specify the default functionality of an object and let its sub-classes to explicitly implement that functionality. Thus, it stands as an abstraction layer that must be extended and implemented by the corresponding sub-classes.

A sample example of using an abstract class is the following. We declare an abstract class, called Instrument:

Instrument.java

12345 abstract class Instrument { protected String name; abstract public void play();}

As we can observe, an Instrument object contains a field name and a method called play, that must be implemented by a sub-class.

Next, we define a sub-class called StringedInstrument that extends the Instrument class and adds an extra field called numberOfStrings:

StringedInstrument.java

123 abstract class StringedInstrument extends Instrument { protected int numberOfStrings;}

Finally, we add two more classes that implement the functionality of a StringedIntrument, called ElectricGuitar and ElectricBassGuitar accordingly. The definition of these newly added classes is shown below:

ElectricGuitar.java

0102030405060708091011121314151617181920 public class ElectricGuitar extends StringedInstrument { public ElectricGuitar() { super(); this.name = "Guitar"; this.numberOfStrings = 6; } public ElectricGuitar(int numberOfStrings) { super(); this.name = "Guitar"; this.numberOfStrings = numberOfStrings; } @Override public void play() { System.out.println("An electric " + numberOfStrings + "-string " + name + " is rocking!"); }}

ElectricBassGuitar.java

0102030405060708091011121314151617181920 public class ElectricBassGuitar extends StringedInstrument { public ElectricBassGuitar() { super(); this.name = "Bass Guitar"; this.numberOfStrings = 4; } public ElectricBassGuitar(int numberOfStrings) { super(); this.name = "Bass Guitar"; this.numberOfStrings = numberOfStrings; } @Override public void play() { System.out.println("An electric " + numberOfStrings + "-string " + name + " is rocking!"); }}

Finally, we create a new class called Execution that contains a single main method:

Execution.java

01020304050607080910111213141516171819 import main.java.music.ElectricBassGuitar;import main.java.music.ElectricGuitar;public class Execution { public static void main(String[] args) { ElectricGuitar guitar = new ElectricGuitar(); ElectricBassGuitar bassGuitar = new ElectricBassGuitar(); guitar.play(); bassGuitar.play(); guitar = new ElectricGuitar(7); bassGuitar = new ElectricBassGuitar(5); guitar.play(); bassGuitar.play(); }}

In this example, we create two different instances of an ElectricGuitar and an ElectricBassGuitar classes and we call their play methods. A sample execution of the aforementioned main method is shown below:

An electric 6-string Guitar is rocking! An electric 4-string Bass Guitar is rocking! An electric 7-string Guitar is rocking! An electric 5-string Bass Guitar is rocking!

1.1 The Purpose of Abstract Classes

The purpose of abstract classes is to function as base classes which can be extended by subclasses to create a full implementation.

Due to this feature, an Abstract Class is being used in scenarios where a complete implementation can be further broken down into a set of repetitive steps with few unique steps.

One such example will be, of URL Processor application which extract the HTML of the website hosted at the URL.

URLProcessorBase.java

package URLProcessor; import java.io.IOException; import java.io.InputStream; import java.net.URL; import java.net.URLConnection; public abstract class URLProcessorBase { public void process(URL url) throws IOException { URLConnection urlConnection = url.openConnection(); InputStream input = urlConnection.getInputStream(); try { processURLData(input); } finally { input.close(); } } protected abstract void processURLData(InputStream input) throws IOException; }

This is a base class that contains the methods used for the implementation of the URL processor.

Now we can have the multiple subclasses of this base class based on the kind of processing that needs to be done on the website hosted at the URL provided as input.

In our example code, we are extracting the HTML passed as the response when we call the URL.

URLProcessorImpl.java

package URLProcessor; import java.io.IOException; import java.io.InputStream; public class URLProcessorImpl extends URLProcessorBase { @Override protected void processURLData(InputStream input) throws IOException { int data = input.read(); while (data != -1) { System.out.print((char) data); data = input.read(); } } }

Now the main driver code.

Driver.java

package URLProcessor; import java.io.IOException; import java.net.URL; public class Driver { public static void main(String[] args) throws IOException { System.out.println("Driver Code for URL Processor Application using Abstract Class Method"); System.out.println("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); System.out.println(); URLProcessorImpl urlProcessor = new URLProcessorImpl(); urlProcessor.process(new URL("https://abhiit89.github.io/")); System.out.println(); System.out.println("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); } }

The output of the driver program is shown below.

abstraction - Java Abstract Class - Driver.java

Output of Driver.java

If we apply a new implementation of the URLProcessorBase class, as shown below.

URLProcessorImplToExtractDataLength.java

package URLProcessor; import java.io.IOException; import java.io.InputStream; public class URLProcessorImplToExtractDataLength extends URLProcessorBase { @Override protected void processURLData(InputStream input) throws IOException { byte[] data = input.readAllBytes(); System.out.print(data.length); } }

URLProcessorImplToExtractDataLength.java is another implementation of the base class URLProcessorBase.java which calculates the length alone of the retrieved data. This implementation in a way enforces the SRP (Single Responsibility Principle).

We can override the original driver class to include this new implementation, but for the sake of clarity, a new driver class called the LengthDriver.java is created.

LengthDriver.java

package URLProcessor; import java.io.IOException; import java.net.URL; public class LengthDriver { public static void main(String[] args) throws IOException { System.out.println("Driver Code for URL Processor Application to Extract length of Retrieved Data using Abstract Class Method"); System.out.println("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); System.out.println(); URLProcessorImplToExtractDataLength metaProcessor = new URLProcessorImplToExtractDataLength(); metaProcessor.process(new URL("https://abhiit89.github.io/")); System.out.println(); System.out.println("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); } }

The output of LengthDriver.java is shown in the snapshot below.

Java Abstract Class - abstraction - LengthDriver.java

The output of LengthDriver.java

This covers the examples related to the Abstract Class.

2. Java Abstract Class vs Interface

In this section, we are going to talk about abstract class vs interface implementation. Java provides and supports the creation of abstract classes and interfaces.

As we stated at the beginning, we can achieve the abstraction by using either abstract classes or interfaces. Both implementations share some common features, but they differ in the following features:

You can learn more about the abstract class vs interface implementation by reading this example.

3. Download the Eclipse Project

That was a tutorial about the Java Abstract Class.

Last updated on Mar. 18th, 2020

Photo of Sotirios-Efstathios Maneas

Sotirios-Efstathios (Stathis) Maneas is a PhD student at the Department of Computer Science at the University of Toronto. His main interests include distributed systems, storage systems, file systems, and operating systems.