kotlin 扩展类的功能
最近有关Kotlin的Reddit的一个问题引起了我的注意:
为什么为自己拥有的类编写扩展功能很不好?
https://bit.ly/37RBZTW
答案之一是,事实恰恰相反:一个好的做法,原因有很多。 其中有两个非常重要的改进了代码设计的东西:
- 可以在可为空的函数上调用该函数
- 对于具有泛型类型的类,添加一个仅在使用特定类型时可用的函数, 例如
Iterable<Int>.sum()
我完全同意这是一个很好的做法。 在本文中,我想进一步详细介绍这两种情况下扩展功能的好处。 我假设您已经熟悉扩展功能。 否则,请参阅相关文档 。
可空友好
为了详细说明第一个用例,让我们考虑String
类中定义的任何函数。 这是在可空类型上调用该函数的方式:
varmyNullableString:String?=null
// Do something on myNullableString
// At this point, myNullableString can be null or not
valcapitalized:String?=myNullableString?.capitalize()
由于myNullableString
可能为null
,因此直接使用standard调用该函数是null
的.
运算符:编译器会将此类代码标记为无效并阻止编译。 要解决编译错误,应使用提供的安全调用运算符 ?.
。
多亏了后者,在运行时,如果变量不为 null
,一切都会按预期进行。 如果是这样,则capitalized
将为null
而不抛出异常。 如果一个人接受沿流传播null
值,这可能就足够了,但是如果没有呢? 是否可以提供默认值(如果该值为null
不是很好,就像Java的Optional.getOrElse()
吗? Java的方法是将逻辑封装在一个类中。 否则,我们将需要一个静态函数。 在Kotlin中,我们可以编写顶级功能:
funorElse(string:String?,default:String)=string?:default
但是,这也可以用其他方式编写。 必须记住,扩展功能只是静态功能的语法糖。 因此,可以在可为null的类型上编写扩展功能 。 让我们牢记上面的功能:
funString?.orElse(default:String)=this?:default
现在可以在可为空的类型上调用orElse()
扩展函数:
valnonNull:String=myNullableString.orElse("")
Kotlin的stdlib提供了String?.orEmpty()扩展功能,这是一种自以为是的选择:默认为空字符串。 同样,为集合提供了一个扩展功能:Collection <T> ?. orEmpty()。
泛型依赖
现在介绍扩展功能的另一个好处。 让我们从一个简单的用例开始:计算属于Person
集合的所有项目的平均年龄。 使用Java的Stream API,解决方案代码如下所示:
valaverage=persons.stream()
.mapToInt(Person::getAge)
.average()
熟悉Java的Stream API的读者知道, Stream
接口上没有方法average()
。 实际上,拥有一个没有任何意义Person
或String
集合的平均值是多少? 但是,在int
的集合上放置一个确实是有意义的。 因此,Java为int
类型提供了专用的Stream
类: IntStream
。 该接口具有在其姊妹接口Stream
找不到的方法。 如上所示,为了将Stream
转换为IntStream
,应使用专用的mapToInt()
方法。
但是,Kotlin允许对泛型类型使用扩展功能。 因此,编写这样的函数而不是创建专门的接口很简单:
funCollection<Int>.average()=
reduce{a,b->a+b} (1)
.toDouble()
.div(size.toDouble())
-
a
和b
的类型为Int
,因此可以一个一个地累积它们
请注意,Kotlin的stdlib提供了Iterable<Int>.sum()
扩展功能。 因此,可以通过替换上面的代码来简化上面的代码:
funCollection<Int>.average()=
sum()
.toDouble()
.div(size.toDouble())
结论
与Java相比,Kotlin提供了更多有关API设计的功能。 特别地,扩展功能提供了非常有趣的选项。 与不好的做法相反,在自己的课堂上使用它们是一个很大的好处。
翻译自: https://blog.frankel.ch/write-extension-functions-own-classes-kotlin/
kotlin 扩展类的功能