Kotlin - 引用操作符 ::

一、概念

将类、函数、属性、构造函数等程序元素转化为可传递的引用对象,就像给这些元素贴上标签,方便在代码中灵活调用。

1.1 使用场景

不能引用局部函数、匿名函数和 Lambda 表达式 ,这些都属于不可引用的实体,如果强行引用,编译时会报错。

|--------|------------------------------------------------------------------|
| 类引用 | 很多时候需要传入 Java 的 Class 对象,通过【类名::class.java】来转换。 |
| 函数引用 | 将函数作为参数传递或赋值给变量,实现函数式编程的高阶特性。极大地增强了代码的灵活性与可扩展性,方便进行函数组合、策略模式等设计。 |
| 属性引用 | 获取属性的 getter 或 setter,实现对属性的间接访问与操作。这在数据绑定、属性监听等场景中十分实用。 |
| 构造函数引用 | 创建对象的工厂函数,便于根据不同需求动态创建对象。在对象池管理、依赖注入等场景中发挥重要作用。 |

1.2 KClass

通过【类名::class】获取 Kotlin 特有的类描述对象,它提供了更丰富的元信息,如simpleName(类名)、qualifiedName(完整类名)、isData(是否为数据类)等,这些额外信息使得在 Kotlin 中进行反射操作和类型处理更加便捷和强大,充分体现了 Kotlin 在语言设计上对开发者的友好和对现代编程需求的支持。

二、使用

2.1 类引用(类名::class.java)

Kotlin 复制代码
retrofit.create(ApiService::class.java)

2.2 函数引用

2.2.1 静态引用(类名::函数名)

静态引用在调用时,需要传入一个接收者对象。

Kotlin 复制代码
class Calculator {
    fun add(a: Int, b: Int): Int = a + b
}
val unboundAdd: Calculator.(Int, Int) -> Int = Calculator::add
val calculator = Calculator()
val result = unboundAdd(calculator, 2, 3)
println(result)  // 输出5

2.2.2 绑定引用(实例名::函数名)

绑定引用已经关联到了特定的实例,在调用时不需要再重复传递接收者对象。

Kotlin 复制代码
class Calculator {
    fun add(a: Int, b: Int): Int = a + b
}
val calculator = Calculator()
val boundAdd: (Int, Int) -> Int = calculator::add
val result = boundAdd(2, 3)
println(result)  // 输出5

2.2.3 顶层函数绑定引用(::函数)

Kotlin 复制代码
fun isEven(number: Int): Boolean = number % 2 == 0
val numbers = listOf(1, 2, 3, 4, 5, 6)
val evens = numbers.filter(::isEven)
println(evens)  // 输出[2, 4, 6]

2.3 属性引用(类名::属性名)

Kotlin 复制代码
class Person(
    val name: String,
    var age: Int
)

val nameGetter: (Person) -> String = Person::name
val person = Person("Alice", 30)
println(nameGetter(person))  // 输出Alice

val ageSetter: (Person, Int) -> Unit = Person::age
ageSetter(person, 31)
println(person.age)  // 输出31

2.4 构造函数引用(::类名)

java 复制代码
class User(val name: String)

val userFactory: (String) -> User = ::User
val newUser = userFactory("Bob")
println(newUser.name)  // 输出Bob

三、优化调用场景

3.1 简化 Lambda 表达式,提升可读性

函数引用是对已存在的具名函数的引用,不会像 Lambda 每次使用时都创建匿名内部类对象,从而减少了对象创建的开销。

Kotlin 复制代码
val strings = listOf("apple", "banana", "cherry")
//一般使用
val lengths = strings.map { it.length }
//优化后使用
val lengths = strings.map(String::length)

3.2 反射

Kotlin 复制代码
class Animal {
    fun makeSound() { println("Some sound") }
}
Kotlin 复制代码
fun main() {
    val className = "Animal"
    val kClass = Class.forName(className).kotlin
    val instance = kClass.constructors.first().call()
    val method = kClass.memberFunctions.find { it.name == "makeSound" }
    method?.call(instance)
}
相关推荐
DogDaoDao1 小时前
Android 硬件编码器参数完全指南:MediaCodec 深度解析
android·音视频·视频编解码·h264·硬编码·视频直播·mediacodec
JohnnyDeng942 小时前
Android 自定义 View:Canvas 绘图与事件分发深度解析
android
Android小码家5 小时前
Framework之Launcher小窗开发
android·framework·虚拟屏·小窗
赏金术士5 小时前
第七章:状态管理实战与架构总结
android·ui·kotlin·compose
颂love6 小时前
MySQL的执行流程
android·数据库·mysql
云起SAAS11 小时前
抖音小游戏源码 - 消消乐 | 含激励广告+成就系统 | 开箱即用商业级消除游戏模板
android·游戏·广告联盟·看激励广告联盟流量主·抖音小游戏源码 - 消消乐
大貔貅喝啤酒12 小时前
基于Windows下载安装Android Studio 3.3.2版本教程(2026详细图文版)
android·java·windows·android studio
程序员码歌12 小时前
OpenSpec 到 Superpowers:AI 编码从说清到做对
android·前端·人工智能
2501_9151063213 小时前
深入解析无源码iOS加固原理与方案,保护应用安全
android·安全·ios·小程序·uni-app·cocoa·iphone
黄林晴16 小时前
重磅官宣:Android UI 开发正式进入 Compose-first 时代
android·google io