8.2. Interfaces¶
An interface is a formal construction within Java. Interfaces allow us to create a set of behaviors that a class must fulfill. A Java interface may contain:
Constants (that is,
static final
fields)Method signatures
Static methods
Default methods
8.2.1. Creating an Interface¶
8.2.1.1. Method Signatures¶
One really useful aspect of using an interface is the ability to specify method signatures. A method signature includes the name, parameters, and return type of a method, but no body.
As part of our cat application, let’s create a method signature, eat
, as part of an interface, Feedable
.
Example
1 2 3 4 5 6 | public interface Feedable
{
void eat();
}
|
In the code above, notice the following:
That we need to use the
interface
keyword to define our interface,Feedable
.eat
only has a signature. We are NOT allowed to provide a body for methods defined in interfaces.eat
also does not have an access modifier. Interface members are alwayspublic
, and while we may use thepublic
modifier, it’s unnecessary. An interface method may NOT have an access modifier that is more restrictive thanpublic
.The
Feedable
interface itself is declaredpublic
, which means any other class may use it. We may also leave offpublic
, making the interface package-private, or usable only within the same package.The name is indicative of the behavior that the interface is intended to describe. While this is only a convention, most interfaces have names that are adjectives. Whatever you do, use meaningful names!
8.2.1.2. Static Methods¶
A static method in an interface can contain code in the body. However, a static method cannot contain any references to instance properties in other classes. This means that our static methods should only deal with universal behaviors that are NOT dependent on instance properties, but may be dependent on constants.
8.2.1.3. Default Methods¶
A default method has a body and is a
fully-formed method. It is preceded with the default
keyword, and it
may be overridden by classes implementing the interface.
1 2 3 4 5 6 7 8 9 | public interface MyInterface {
void someMethod();
default void someOtherMethod() {
// ...code goes here...
}
}
|
The intended purpose of default methods is to allow programmers to add a method to an interface that has already been released, while not forcing those already using the interface to add new code to their classes. You should avoid using default methods in all situations other than the one described here.
8.2.2. Implementing an Interface¶
The purpose of an interface is to define a contract of behaviors that classes uphold. In doing so, we say that they “implement the
interface”. The syntax for implementation is similar to that for
inheritance. Here’s how we can use the Feedable
interface in
defining our Cat
class.
Example
1 2 3 4 5 6 7 8 9 10 11 12 | public class Cat implements Feedable
{
@Override
public void eat()
{
// method implementation
}
// ...rest of the class definition...
}
|
Since we’ve declared that Cat
implements Feedable
, we have to
provide an implementation for the eat
method, with the signature as
specified in the interface definition. Note that we use @Override
,
just as we do when overriding an inherited method in a subclass. Like
that situation, using @Override
when implementing methods defined in
an interface will enable the compiler to check that your method
signature does indeed match that of the interface.
Note
You may both extend a class and implement an interface at the same time:
1 2 3 4 | public class MyClass extends MySubclass implements MyInterface
{
// ...code...
}
|
As with classes, interfaces define a type that can be used when
declaring fields, parameters, and local variables. This allows us to relax the requirements on our code
elsewhere, thus making it more extensible and adaptable.
If an application is extensible, it is easier for programmers for new capabilities to be added later on. For example,
here’s how we might modify our CatOwner
class:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | public class CatOwner
{
private Feedable pet;
public CatOwner(Feedable pet) {
this.pet = pet;
}
public void feedTheCat() {
// ...code to prepare the cat's meal...
pet.eat();
}
}
|
Note that we’ve declared the property pet
to be of type
Feedable
. This class assumes that the only behavior of pet
that
we’ll need within the class is the ability to eat
. But if that’s all
we need, then we should relax the requirements on the pet
property
as much as possible. In fact, there’s nothing specific about cats in
this class, so we might make our code a step more abstract and flexible
by doing the following:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | public class PetOwner
{
private Feedable pet;
public PetOwner(Feedable pet) {
this.pet = pet;
}
public void feedThePet() {
// ...code to prepare the pet's meal...
pet.eat();
}
}
public class CatOwner extends PetOwner
{
// code that requires Cat-specific behavior
}
|
We’ve created a PetOwner
class that encapsulates the behavior for any pet (any Feedable
, actually), and have
CatOwner
extend PetOwner
. This allows other classes to extend
PetOwner
to make, say, a DogOwner
that knows how to play fetch
with their pet, or a HorseOwner
that knows how to ride their pet. It
also reduces the dependency of the feedThePet
method on the specific
type of pet, since it doesn’t need to care.
To use this new class design, we can revise the sample code from above as follows:
1 2 3 4 | HouseCat suki = new HouseCat("Suki", 12);
CatOwner Annie = new CatOwner(suki);
Annie.feedThePet();
|
While the code usage here remains unchanged except for changing the
method name from feedTheCat
to the more generic feedThePet
, the
opportunities for using the classes we’ve built are much wider since the
defined classes are no longer dependent on the specific Cat
class.
Also notice that we’ve used the object suki
in a polymorphic way,
creating it as a HouseCat
, but using it as a Feedable
within the
CatOwner
class.
Note
Like inheritance, interfaces enable polymorphic usage of objects. We can create an object and then use it in different contexts based on the interfaces that it implements.
Interfaces may not be created like objects are, with new
.
You may implement an interface, or declare variables and parameters as
interface types. You can not, however, create an instance of an
interface.
8.2.3. Benefits of Using Interfaces¶
Once you get used to interfaces, you’ll begin to think more abstractly about which behaviors your code requires rather than which classes your code requires. This means you’ll be able to “code to interfaces” (an OOP principle) instead of coding to classes, and your code will become more flexible and extensible.
Here are a few benefits of using interfaces:
You can only extend one class, but you may implement many interfaces.
You can extend a class and implement an interface at the same time.
By declaring variables and parameters as interface types, you make your code useful for a much wider variety of situations.
When you declare properties and return types to be interface types, you decouple code using your classes from the actual class types you use. This means that you are free to change the specific implementation of your classes without affecting those using them.
Remember that you don’t need to start creating interfaces to use their power! As we will cover later in the chapter, when working with collections, we can declare variables and parameters to be interface types if we only need to use specific behaviors such as ordering or iteration.
8.2.4. Check Your Understanding¶
Question
Choose the appropriate option to fill in the blanks.
A class can extend _______ class(s) and implement ________ interface(s).
one, one
one, more than one
more than one, one
more than one, more than one
Question
True or False:
An interface can contain a static
variable, a final
variable, and a static final
variable.
Question
What keywords are missing from the following interface?
1 2 3 4 5 6 | public Temperature {
double final ABSOLUTE_ZERO = -273.15;
double convertTemp();
}
|