Abstract Classes vs. Interfaces
First published at Tuesday 2 October 2012
Warning: This blog post is more then 13 years old – read and use with care.
Abstract Classes vs. Interfaces
Features of object oriented languages are often use from a purely technical perspective, without respect to their actual semantics. This is fine as long as it works for you, but might lead to problems in the long run. In this article I discuss the semantical differences between abstract classes and interfaces
. I also outline why following the semantics of those language constructs can lead to better code.
Disclaimer: This is of course kind of artificial. If using the language constructs in other ways works for you, it's fine. It can make communication, extension of code and maintenance of code harder, though.
Definitions
First, we need to differ between interface
and interface. When typeset in mono space I mean the language construct -- otherwise I talk about the public methods of any object / class. The second one often is also called "public interface", "class interface" or even "class signature".
Classes are Types
A class denotes a type.
We know that objects consist of internal state and methods to operate on this state. The interactions between objects are implemented by calling methods on other, related objects. Those methods might, or might not, modify the internal state of the called object.
Classes, as they define the blueprints for those objects, define the type of objects, thus we know how to use an object. When talking about types there are some natural implications, which apply especially when extending from another class, which means to create a sub-type. One is the Liskov Substitution Principle:
“Functions that use pointers or references to base classes must be able to use objects of derived classes without knowing it.”
This basically translates to: "Do not mess up the interface". It is required, so you can use a sub-type (derived class) without any special knowledge, just by reading the documentation / interface of the parent class. Imagine a Cache
class, from which you extend a Cache\FileSystem
, Cache\Memcached
and maybe a Cache\Null
. You do not want to implement any kind of special handling in every place where an object of any of those derived classes is used in your application. All of them should just behave like a Cache
. Keep this in mind.
If we can use any sub-type of a type just as the parent type defines, we talk about Subtype Polymorphism. Something you definitely want to achieve. Abstract classes are naturally used when you need such a parent type, which requires some specialization, like the Cache
class mentioned above. I'll get back to why it makes no sense to extract an interface
from this. And yes, you are allowed to define abstract classes, even if you don't intend to implement any code already and don't have any properties to define.
interface
An interface
describes an aspect of the interface of a type.
PHP does not support multiple inheritance, but it allows you to implement any number of interfaces, right? The reason for this is, that an interface
annotates usage aspects on the interface of a type. The Interface Segregation Principle also claims:
“Clients should not be forced to depend upon interfaces that they do not use.”
This means that a type using another type should only depend on an interface, which is actually used. Depending on a god class, for example, is the exact opposite. You would not know which methods are actually called from looking at the injection signature. But this also implies that the interfaces should be designed small and task-specific.
To stay in the context of the example mentioned above, a proper interface would be Cacheable
to annotate on certain classes that its instances can be cached. The natural counterpart of a Cache
, if you want to handle objects. This is a minimal usage aspect of a class and other classes requiring this interface
will most likely, and may only, use the methods required by this interface
. Implementors of this interface would also not be forced to implement meaningless methods.
Telling Apart
Those definitions are nice, and fluffy -- but what happens if you define an interface
instead of using an abstract class? (This is at least what I see most.)
You invite others to create god classes
Interfaces do not allow to already provide code or define properties. Their only "benefit" over abstract classes is that you can implement multiple of them in a class / type.
So why would you define a
Cache
interface. Obviously you do not want that some class implements theCache
logic while also being aLogger
and aORM
-- even all might access the same storage.While this, in theory, provides more "flexibility" for the future, every usage of this flexibility would hurt your software design -- a lot.
You overload interfaces
A common developer pattern is to define a type, and then extract one single
interface
from the class interface. Such interfaces do not define a single usage aspect any more. If someone wants to implement one aspect of thisinterface
and two or three aspects from some other "interfaces
" it would result in one really big class with dozens of empty methods. Or, even worse, dozens of implemented methods, which are never used. This would obviously be wrong.
But...
There is this one popular quote:
“Code against interfaces, not implementations.”
"Interface" is not typeset in mono space. Look it up!
To phrase it better, like it is done in the Dependency Inversion Principle: “Depend on abstractions, not on concretions.”. "Interfaces" in the above quote may very well be abstract classes. Don't forget about those.
Examples & Hints
Classic examples for proper interfaces would be:
Cacheable
Serializeable
I have two mnemonics for people I discuss Object Oriented Design with:
interface
names should end withable
oring
.This is obviously not always true. But ending on
able
is a strong indicator that such aninterface
just annotates one usage aspect.interfaces
make sense to be implemented in entirely different types.An
interface
usually makes sense to be implemented in types, which otherwise have barely anything in common. A prime example again would beCacheable
orCountable
, which even is a properinterface
defined in the SPL.To come up with a "real-world" example: Let's consider an
interface
Drinkable
. You could implement that one on cups, on the sea and maybe even on humans, if there are vampires around. Otherwise humans, seas and cups probably do not have that much in common.
Examples for proper abstract classes could be:
Log(ger)
Cache
Again I have a mnemonic to help you with the decision:
An implemented abstract class can stand on its own.
If you implement all remaining abstract methods of an abstract class, you get a proper member in your world. It does not require any additional methods to be usable by others. It probably will require some dependencies (like a storage), but most other objects will happily just call the methods provided by the abstract class.
Final hint:
An interface
must not define a constructor. Never. The same is true for most abstract classes. By defining a constructor you predefine and limit what dependencies may be injected in implementations. It is very likely to change for different implementations. But this is actually a topic which deserves its very own blog post.
tl;dr
An interface
annotates a usage aspect of a type / class. Do not just extract interfaces from classes. Use abstract classes, when defining a base type.
Subscribe to updates
There are multiple ways to stay updated with new posts on my blog: