ProAndroidDev

The latest posts from Android Professionals and Google Developer Experts.

Follow publication

[Kotlin Pearls 5] Multiple Inheritance

Uberto Barbini
ProAndroidDev
Published in
7 min readApr 26, 2019

Another kind of multiple inheritance…
interface ApiClient {
fun callApi(request: String): String
}
//implementation for testing as singleton
object ClientMock: ApiClient {
override fun callApi(request: String): String = "42"
}
//implementation real as a class
class HttpApiClient: ApiClient {
override fun callApi(request: String): String {
//... lot of code
return "The Answer to $request is 42"
}
}
class DoSomething: ApiClient by HttpApiClient(){
//other stuff needed to do something
}
class DoSomethingTest(): ApiClient by ClientMock {
//other stuff needed to do something
}
Click on the button Decompile to see the Java equivalent code
public final class DoSomething implements ApiClient {
// $FF: synthetic field
private final HttpApiClient $$delegate_0 = new HttpApiClient();
@NotNull
public String callApi(@NotNull String request) {
Intrinsics.checkParameterIsNotNull(request, "request");
return this.$$delegate_0.callApi(request);
}
}
class DoSomethingDelegated(client: ApiClient): ApiClient by client {
//other stuff needed to do something
}
class DoSomethingOverriden( client: ApiClient): ApiClient by client {
override fun callApi(request: String): String {
return "something different $request"
}
}
class DoSomethingWrapper(private val client: ApiClient): ApiClient by client {    override fun callApi(request: String): String {
println("DoSomethingWrapper: hello!")
return client.callApi(request)
}
}
class DoSomethingDoubleBy( client: ApiClient): ApiClient by object : ApiClient by client {
override fun callApi(request: String): String {
println("DoSomethingDoubleBy: hello!")
return client.callApi(request)
}
}
data class Image (val name: String)interface Printer{
fun turnOn()
fun isOn(): Boolean
fun printCopy(image: Image)
}
interface Scanner{
fun turnOn()
fun isOn(): Boolean
fun scan(name: String): Image
}
object laserPrinter: Printer {
var working = false
override fun isOn(): Boolean = working override fun turnOn() {
working = true
}
override fun printCopy(image: Image) {
println("printed ${image.name}")
}
}object fastScanner: Scanner {
var working = false
override fun isOn(): Boolean = working override fun turnOn() {
working = true
}
override fun scan(name: String): Image = Image(name)}class ScannerAndPrinter(scanner: Scanner, printer: Printer): Scanner by scanner, Printer by printer { override fun isOn(): Boolean = (this as Scanner).isOn() override fun turnOn() = (this as Scanner).turnOn()

fun scanAndPrint(imageName: String) = printCopy(scan(imageName))
}
typealias Row = Map<String, Any> //a db row is expressed as a Map field->valueinterface DbConnection {
//abstraction on the db connection/transaction etc
fun executeQuery(sql: String): List<Row>
}
interface SqlRunner<out T> { //out is because we can return T or subtypes
// interface to execute sql statement and return domain objects
fun builder(row: Row): T
fun executeQuery(sql: String): List<Row>
}
//declared outside SqlRunner interface to avoid overriding and multiple implementations
fun <T> SqlRunner<T>.fetchSingle(sql: String): T = builder( executeQuery(sql).first() )
fun <T> SqlRunner<T>.fetchMulti(sql: String): List<T> = executeQuery(sql).map { builder(it) }
//a real example would be: class JdbcDbConnection(dbConnString: String): DbConnectionclass FakeDbConnection(): DbConnection{
//trivial example but in reality manage connection pool, transactions etc and translate from JDBC
override fun executeQuery(sql: String): List<Row> {
return listOf( mapOf("id" to 5, "name" to "Joe") )
}
}//now how to use all this for retrieve Usersinterface UserPersistence{
//interface needed by the domain
fun fetchUser(userId: Int): User
fun fetchAll(): List<User>
}
class UserPersistenceBySql(dbConn: DbConnection): UserPersistence, SqlRunner<User> by UserSql(dbConn) {
//translate domain in sql statements but still abstract from db connection,transactions etc.
override fun fetchUser(userId: Int): User {
return fetchSingle("select * from users where id = $userId")
}
override fun fetchAll(): List<User> {
return fetchMulti("select * from users")
}
}
class UserSql( dbConn: DbConnection) : SqlRunner<User>, DbConnection by dbConn {
// implementation for User
override fun builder(row: Row): User = User(
id = row["id"] as Int,
name = row["name"] as String)
// note that we don't need to implement executeQuery because is already in DbConnection
}
object UserRepository: UserPersistence by UserPersistenceBySql(FakeDbConnection())//example of use
fun main() {
val joe = UserRepository.fetchUser(5)
println("fetched user $joe")
}

Published in ProAndroidDev

The latest posts from Android Professionals and Google Developer Experts.

Written by Uberto Barbini

JVM and Kotlin independent consultant. Passionate about Code Quality and Functional Programming. Author, public speaker and OpenSource contributor.

Responses (3)

Write a response

i think your title is very misleading. there is no ‘multiple inheritance’ going on here, and i think your article generally hints at that. it would be much clearer to explicitly state that ‘by’ is for delegation pattern instead.

--

What a brillient example! You have really cleared my fundamental of delegation. Thank you.

--