Kotlin责任链模式与泛型

key*_*rai 2 design-patterns chain-of-responsibility kotlin

使用责任链模式我遇到了一个问题,其中下一个链元素应该具有与第一个元素相同的泛型类型.我知道为什么会发生这种情况:第一个处理程序期望第二个处理程序使用泛型类型"Apple".我只是不知道如何解决它.

这里有一个关于如何在java中处理它的答案,但由于java没有具体的类型,而且所有这些方法在Kotlin中应该看起来不同,对吧?

我想到了不同的选择:

  1. 不要使用泛型 - 会导致将集合类型转换为特定的子类型,并且看起来不干净
  2. 尝试使用具体类型(如何?)

为了说明问题,我将在下面发布一个演示代码.


data class Apple(val name:String, val color:Int) 
data class Orange(val circumference:Double)

object Main{
    @JvmStatic
    fun main(args: Array<String>) {
        val first = FirstHandler()
        val second = SecondHandler()
        first.setNextHandler(second)  // !!! wrong type here since <Apple> is expected
        first.process()
    } 
}

abstract class ChainHandler<T>{
    protected var nextHandlerInChain:ChainHandler<T>? = null
    fun setNextHandler(handler: ChainHandler<T>): ChainHandler<T> {
        this.nextHandlerInChain = handler
        return handler
    }

    abstract fun peel(): Collection<T>
    abstract fun process():MutableMap<String,Any> }

class FirstHandler : ChainHandler<Apple>() {
    override fun peel(): Collection<Apple> {
        return Collections.emptyList<Apple>()
    }
    override fun process(): MutableMap<String, Any> {
        val peeledApples = peel()
        val map = nextHandlerInChain?.process()
        map?.put("apples",peeledApples) ?:kotlin.run {
            val map = mutableMapOf<String,Any>()
            map.put("apples",peeledApples)
        }
        return map!!
    } }

class SecondHandler : ChainHandler<Orange>() {
    override fun peel(): Collection<Orange> {
        return Collections.emptyList<Orange>()
    }
    override fun process(): MutableMap<String, Any> {
        val peeledOranges = peel()
        val map = nextHandlerInChain?.process()
        map?.put("oranges",peeledOranges) ?:kotlin.run {
            val map = mutableMapOf<String,Any>()
            map.put("oranges",peeledOranges)
        }
        return map!!
    } 
}
Run Code Online (Sandbox Code Playgroud)

mar*_*ran 5

Kotlin有一种叫做星形投影的东西,可以帮助你.它基本上告诉编译器你并不关心ChainHandler你得到什么类型.您可以使用它来进行setNextHandler编译,如下所示:

abstract class ChainHandler<T>{
  // Note the star projection here
  protected var nextHandlerInChain: ChainHandler<*>? = null

  // Defining a type parameter S, so that the return type is equal to the input type.
  fun <S> setNextHandler(handler: ChainHandler<S>): ChainHandler<S> {
    this.nextHandlerInChain = handler
    return handler
  }

  ...
}
Run Code Online (Sandbox Code Playgroud)

您可以在此处阅读有关星级预测的更多信息:https://kotlinlang.org/docs/reference/generics.html#star-projections

至于reifying类型参数:Reified类型参数仅适用于内联函数.不适用于类的类型参数.