You might have already heard about the delegation pattern or at least used it without even knowing it had a name. It allows a type to forward one or more of its method calls to a different type. Therefore, you need two types to achieve this—the delegate and the delegator.
This might sound like it's a proxy pattern, but it isn't. A proxy pattern is meant to provide a placeholder for an instance to get full control while accessing it. Let's say you are writing a UI framework and you start where your abstraction is UIElement. Each of the components defines a getHeight and getWidth. Consider the following diagram:
The following code block shows you the unified modeling language (UML) translated into Kotlin. We defined the UIElement interface with both the Panel and Rectangle classes inheriting the following:
interface UIElement {
fun getHeight(): Int
fun getWidth(): Int
}
class Rectangle(val x1: Int, val x2: Int, val y1: Int, val y2: Int) : UIElement {
override fun getHeight() = y2 - y1
override fun getWidth() = x2 - x1
}
class Panel(val rectangle: Rectangle) : UIElement by rectangle
val panel = Panel(Rectangle(10,100,30,100))
println("Panel height:"+panel.getHeight())
println("Panel witdh:" + panel.getWidth())
You have probably noticed the by keyword in the Panel class definition. It's basically a hint for the compiler to do the work for you—forwarding the calls for the methods exposed by the UIElement interface to the underlying Rectangle object.
Through this pattern, you replace inheritance with composition. You should always favor composition over inheritance for the sake of simplicity, reducing type coupling, and flexibility. Using this approach, you can choose and swap the type you put in the delegate position based on various requirements.