如何在Kotlin中将零赋给泛型Number?

huangapple go评论59阅读模式
英文:

How to assign a zero to genric Number in Kotlin?

问题

我是 Kotlin 新手,尝试编写一个简单的泛型方法,但它无法编译。

  1. inline fun <T : Number> List<T>.firstOrZero(): T {
  2. if (isEmpty())
  3. return 0 // 这行无法编译
  4. else
  5. return this[0]
  6. }

有没有办法解决这个问题?

英文:

I'm new to Kotlin, and I'm trying to write a simple generic method but it doesn't compile

  1. inline fun &lt;T:Number&gt; List&lt;T&gt;.firstOrZero() : T {
  2. if (isEmpty())
  3. return 0 // this line doesn&#39;t compile
  4. else
  5. return this[0]
  6. }

Is there a way to do it?

答案1

得分: 2

你需要为你想要支持的每个Number子类处理它。例如像这样:

  1. inline fun <reified T : Number> List<T>.firstOrZero(): T {
  2. return (if (isEmpty()) when (T::class) {
  3. Int::class -> 0
  4. Double::class -> 0.0
  5. Float::class -> 0f
  6. BigInteger::class -> BigInteger.ZERO
  7. else -> throw RuntimeException()
  8. }
  9. else
  10. this[0]) as T
  11. }
英文:

You will need to handle it for every single Number subclass you want to support. For example like this

  1. inline fun &lt;reified T : Number&gt; List&lt;T&gt;.firstOrZero(): T {
  2. return (if (isEmpty()) when (T::class) {
  3. Int::class -&gt; 0
  4. Double::class -&gt; 0.0
  5. Float::class -&gt; 0f
  6. BigInteger::class -&gt; BigInteger.ZERO
  7. else -&gt; throw RuntimeException()
  8. }
  9. else
  10. this[0]) as T
  11. }

答案2

得分: 1

我们不能以一种完全通用的方式在 Kotlin 中处理这种情况。无论我们选择什么解决方案,都必须分别处理每种类型的 Number

因此,我建议不要使用通用函数,而是使用多个重载:

  1. fun List<Int>.firstOrZero() = if (isEmpty()) 0 else this[0]
  2. fun List<Long>.firstOrZero() = if (isEmpty()) 0 else this[0]

这应该比使用反射的通用解决方案潜在地更快,因为类型在编译时而不是运行时解析。

如果您有更多类似的情况,您的情况更复杂等,而且您担心代码重复,您可以内联公共代码:

  1. fun List<Int>.firstOrZero() = firstOrDefault { 0 }
  2. fun List<Long>.firstOrZero() = firstOrDefault { 0 }
  3. private inline fun <T:Number> List<T>.firstOrDefault(default: () -> T) : T {
  4. if (isEmpty())
  5. return default()
  6. else
  7. return this[0]
  8. }

但对于特定的 firstOrZero() 情况,我认为这有点过于复杂。

英文:

We can't really handle this case in a fully generic way in Kotlin. Whatever solution we choose, we have to handle each type of Number separately.

Therefore, I suggest to not go with a generic function, but with multiple overloads:

  1. fun List&lt;Int&gt;.firstOrZero() = if (isEmpty()) 0 else this[0]
  2. fun List&lt;Long&gt;.firstOrZero() = if (isEmpty()) 0 else this[0]

This should be potentially faster than generic solutions using reflection, because the type is resolved at the compile time instead of runtime.

If you have more cases like this, your cases are more complicated, etc. and you are concerned about the code duplication, you can inline the common code:

  1. fun List&lt;Int&gt;.firstOrZero() = firstOrDefault { 0 }
  2. fun List&lt;Long&gt;.firstOrZero() = firstOrDefault { 0 }
  3. private inline fun &lt;T:Number&gt; List&lt;T&gt;.firstOrDefault(default: () -&gt; T) : T {
  4. if (isEmpty())
  5. return default()
  6. else
  7. return this[0]
  8. }

But for the specific case of firstOrZero() I think it is an overkill.

答案3

得分: -3

你需要确保返回类型与预期类型匹配。

因为你定义了泛型类型'T',所以需要返回'T',它是'Number'的子类型。

英文:

You need to ensure that the return type matches the expected type.

Because you defined the generic type 'T', you need to return 'T', which extends 'Number'.

  1. inline fun &lt;T : Number&gt; List&lt;T&gt;.firstOrZero(): T {
  2. if (isEmpty())
  3. return 0 as T
  4. else
  5. return this[0]
  6. }

答案4

得分: -4

有一种方法可以做到这一点。您可以使用默认关键字为 T 类型参数提供默认值。在这种情况下,默认值将为 0。以下代码将编译并运行:

  1. inline fun <T:Number> List<T>.firstOrZero() : T {
  2. if (isEmpty())
  3. return 0.default<T>() // 此行编译
  4. else
  5. return this[0]
  6. }

默认关键字可用于为任何类型参数提供默认值。当您希望确保方法或属性始终具有值时,即使未指定类型参数时,这很有用。

在此示例中,使用默认关键字为 T 类型参数提供了默认值 0。这意味着如果未指定 T 类型参数,该方法将返回 0。

以下是如何使用 firstOrZero() 方法的示例:

  1. val list = listOf(1, 2, 3)
  2. val firstNumber = list.firstOrZero()
  3. println(firstNumber) // 打印 1
  4. val emptyList = emptyList<Int>()
  5. val firstNumberInEmptyList = emptyList.firstOrZero()
  6. println(firstNumberInEmptyList) // 打印 0
英文:

There is a way to do it. You can use the default keyword to provide a default value for the T type parameter. In this case, the default value will be 0. The following code will compile and run:

  1. inline fun &lt;T:Number&gt; List&lt;T&gt;.firstOrZero() : T {
  2. if (isEmpty())
  3. return 0.default&lt;T&gt;() // this line compiles
  4. else
  5. return this[0]
  6. }

The default keyword can be used to provide a default value for any type parameter. This can be useful when you want to make sure that a method or property always has a value, even if the type parameter is not specified.

In this case, the default keyword is used to provide a default value of 0 for the T type parameter. This means that if the T type parameter is not specified, the method will return 0.

Here is an example of how to use the firstOrZero() method:

  1. val list = listOf(1, 2, 3)
  2. val firstNumber = list.firstOrZero()
  3. println(firstNumber) // prints 1
  4. val emptyList = emptyList&lt;Int&gt;()
  5. val firstNumberInEmptyList = emptyList.firstOrZero()
  6. println(firstNumberInEmptyList) // prints 0

huangapple
  • 本文由 发表于 2023年6月12日 14:43:46
  • 转载请务必保留本文链接:https://go.coder-hub.com/76454147.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定