kotlin基础之泛型和委托

embedded/2024/9/25 17:18:13/

Kotlin泛型的概念及使用

泛型概念

在Kotlin中,泛型(Generics)是一种允许在类、接口和方法中使用类型参数的技术。这些类型参数在实例化类、实现接口或调用方法时会被具体的类型所替代。泛型的主要目的是提高代码的复用性、类型安全性和可读性。

泛型使用
  1. 泛型类

定义一个泛型类,可以在类名后面加上尖括号< >,并在其中声明类型参数。

class Box<T>(val item: T) {
fun getContent(): T {
return item
}
}
// 使用时指定类型参数
val intBox = Box<Int>(10)
val stringBox = Box<String>("Hello")
  1. 泛型函数

函数也可以有类型参数。

fun <T> printItems(items: List<T>) {
for (item in items) {
print(item)
print(", ")
}
println()
}
// 使用时,Kotlin会自动推断T的类型
printItems(listOf(1, 2, 3))
printItems(listOf("a", "b", "c"))
  1. 泛型接口

与泛型类和泛型函数类似,接口也可以有类型参数。

interface Listener<T> {
fun onItemClicked(item: T)
}
// 实现泛型接口
class ButtonClickListener<T> : Listener<T> {
override fun onItemClicked(item: T) {
// 处理点击事件
}
}
协变(Covariance)

协变是指在一个泛型类型中,如果类型参数是某个类的子类型,那么使用这个类型参数的泛型类型也应该是父类泛型类型的子类型。在Kotlin中,通过out修饰符实现协变。

interface Source<out T> {
fun next(): T?
}
fun demo(strs: Source<String>) {
// ...
}
val intSource: Source<Int> = ...
// 因为Int是String的子类型(在Kotlin中String不是Int的子类,这里仅作示例),但Source<Int>不是Source<String>的子类型
// 所以不能直接传递intSource给demo函数,但可以通过协变实现
demo(intSource as Source<String>) // 错误:类型不匹配
// 正确的协变用法
val stringSource: Source<out String> = intSource as? Source<out String> // 这里假设intSource实际上可以转换为Source<out String>
if (stringSource != null) {
demo(stringSource) // 正确
}

注意:在Kotlin中,String并不是Int的子类型,上面的例子仅用于说明协变的概念。

逆变(Contravariance)

逆变与协变相反,它指的是在一个泛型类型中,如果类型参数是某个类的父类型,那么使用这个类型参数的泛型类型也应该是子类泛型类型的父类型。在Kotlin中,通过in修饰符实现逆变。

interface Sink<in T> {
fun put(item: T)
}
fun fill(sink: Sink<Number>) {
// ...
}
val stringSink: Sink<String> = ...
// 因为String是Number的子类型,但Sink<String>不是Sink<Number>的子类型
// 所以不能直接传递stringSink给fill函数,但可以通过逆变实现
fill(stringSink as Sink<Number>) // 错误:类型不匹配
// 正确的逆变用法
val numberSink: Sink<in Number> = stringSink as? Sink<in Number> // 这里假设stringSink实际上可以转换为Sink<in Number>
if (numberSink != null) {
fill(numberSink) // 正确
}

同样,上面的例子仅用于说明逆变的概念,实际上String不是Number的子类型。

星号投射(Star Projection)

星号投射(*)在Kotlin中用于处理泛型类型的通配符情况。当你声明一个泛型类型但不想指定具体的类型参数时,可以使用星号投射。

使用方式

  1. 协变星号投射List<out T*> 通常简化为 List<*>。这表示列表中的元素可以是任何类型,但当你从列表中取出元素时,它的类型会被视为Any?(因为任何类型都可以赋值给Any?)。

val list: List<*> = ... // list可以是任何类型的List
for (item in list) {
if (item is String) {
println(item.length) // 只有在确定item是String类型时才能调用其方法
}
}
  1. 逆变星号投射:在Kotlin中,逆变星号投射不常用,因为Kotlin的泛型系统主要基于协变和不变。但在某些高级用法中,你可能会遇到类似于Sink<in T*>的逆变星号投射,这表示该接口或类可以接受任何类型的参数。

委托(Delegation)

概念

委托(Delegation)是一种设计模式,它允许一个对象(委托对象)将其职责的一部分或全部委托给另一个对象(被委托对象)。委托模式可以提高代码的复用性和可维护性。

使用

  1. 类委托:在Kotlin中,可以使用by关键字来实现类委托。这允许一个类将某些方法的实现委托给另一个类的实例。

class Base {
fun printMessage() {
println("Message from Base")
}
}
class Derived(b: Base) : Base() by b {
// Derived类将Base类的printMessage方法委托给b实例
}
fun main() {
val derived = Derived(Base())
derived.printMessage() // 输出 "Message from Base"
}

注意:在上面的例子中,Derived类继承了Base类,但实际上并没有重写printMessage方法。相反,它使用by关键字将该方法的调用委托给了b实例(即Base类的一个实例)。
2. 属性委托:Kotlin还支持属性委托,允许你将属性的getset操作委托给另一个对象或表达式。这可以通过在属性声明中使用by关键字和相应的委托提供程序来实现。

class LazyValue<T>(private val initializer: () -> T) {
private var value: T? = null
fun getValue(): T {
if (value == null) {
value = initializer()
}
return value!!
}
// 这里省略了setValue方法,因为我们只关心只读属性
}
class Example {
val lazyString: String by LazyValue { "Hello, World!" }
}
fun main() {
val example = Example()
println(example.lazyString) // 输出 "Hello, World!",并且只会在第一次访问时计算值
}

在这个例子中,lazyString属性的get操作被委托给了LazyValue类的实例。当第一次访问lazyString时,它会调用LazyValuegetValue方法来计算并缓存值。之后的访问将直接返回缓存的值。


http://www.ppmy.cn/embedded/44043.html

相关文章

4月平板电脑行业线上销售数据分析

由于全球科技发展趋势&#xff0c;如AI技术的应用&#xff0c;以及厂商新品发布计划&#xff1b;同时&#xff0c;平板电脑作为个人电脑的延伸产品&#xff0c;其便携性和生产力相较于手机具有明显优势&#xff0c;这也为行业的进一步发展提供了动力。 据鲸参谋数据统计&#…

程序猿转型做项目经理一定要注意这 5 个坑

前言 国内的信息系统项目经理&#xff0c;很多都是从技术骨干转型的&#xff0c;我就是这样一路走过来的&#xff0c;这样有很多好处&#xff0c;比如技术过硬容易服众、熟悉开发流程更容易把控项目进度和质量、开发过程中碰到难题时更好组织攻坚等等&#xff0c;但是所谓成也…

【微机原理及接口技术】可编程计数器/定时器8253

【微机原理及接口技术】可编程计数器/定时器8253 文章目录 【微机原理及接口技术】可编程计数器/定时器8253前言一、8253的内部结构和引脚二、8253的工作方式三、8253的编程总结 前言 本篇文章就8253芯片展开&#xff0c;详细介绍8253的内部结构和引脚&#xff0c;8253的工作方…

ITIL4认证考试这么贵,还值得考证吗,有必要学吗?

从2023年4月1日开始&#xff0c;ITIL 4是Foundation认证将会捆绑OTM(Official Training Materials),这样在一次ITIL4的考试费中将会捆绑&#xff1a;试卷费电子教材书费监考费OTM费&#xff0c;每一种考试费都相较于2022年有涨幅&#xff0c;再加上PeopleCert收取的授权机构的授…

研发设计管理、研发设计管理系统有哪些

研发设计管理系统种类繁多&#xff0c;每种系统都有其特定的功能和用途。以下是一些常见的研发设计管理系统及其主要功能&#xff1a; PLM&#xff08;产品生命周期管理&#xff09;研发管理系统&#xff1a; 功能&#xff1a;管理产品从概念、设计、开发、制造、销售到维护的…

【MySQL精通之路】SQL优化(1)-查询优化(7)-嵌套循环联接

主博客&#xff1a; 【MySQL精通之路】SQL优化(1)-查询优化-CSDN博客 上一篇&#xff1a; 【MySQL精通之路】SQL优化(1)-查询优化(6)-索引条件下推-CSDN博客 下一篇&#xff1a; 1.嵌套循环连接算法 一个简单的嵌套循环联接&#xff08;NLJ&#xff09;算法一次从循环中的…

LeetCode:78.子集

解答 class Solution:def subsets(self, nums: List[int]) -> List[List[int]]:res[[]]for i in nums:res[[i]num for num in res]return res代码解释 这段代码定义了一个名为Solution的类&#xff0c;并在其中定义了一个名为subsets的方法。该方法接受一个整数列表nums作…

编程中的模块迷宫:区分与正确使用

新书上架~&#x1f447;全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我&#x1f446;&#xff0c;收藏下次不迷路┗|&#xff40;O′|┛ 嗷~~ 目录 一、模块混淆的陷阱 二、碳模块与探母模块的区别 三、如何正确使用模块 四、代码示例 五…