当前位置: 首页 > news >正文

Kotlin 中 infix 关键字的原理和使用场景

在 Kotlin 中,使用 infix 关键字修饰的函数称为中缀函数,使用是可以省略 .(),允许以更自然(类似自然语言)的语法调用函数,这种特性可以使代码更具可读性。

1 infix 的原理

中缀函数必须满足以下条件:

  • 必须是成员函数(成员方法)或扩展函数;
  • 有且仅有一个参数,且不能有默认值;
  • 必须显式声明 infix 关键字;

调用方法:

  • 普通函数:a.function(b);
  • 中缀函数:a function b (省略点号和括号,增强可读性);

示例:

class Person(private val name: String) {// 成员中缀函数infix fun say(message: String) {println("$name says $message")}
}// 扩展中缀函数
infix fun Int.multiply(factor: Int): Int = this * factor

反编译成 Java 代码:

public final class Person {private final String name;public final void say(@NotNull String message) {Intrinsics.checkNotNullParameter(message, "message");String var2 = this.name + " say " + message;System.out.println(var2);}public Person(@NotNull String name) {Intrinsics.checkNotNullParameter(name, "name");super();this.name = name;}
}

底层实现:

  • 中缀函数本质上是普通函数,但通过 infix 关键字允许省略 . 和括号 ()
  • 编译器在语法层面给了支持;(Kotlin 的很多特性都是在语法和编译器上的优化)

2 使用场景

2.1 自然语言风格的 API

适用于需要代码接近自然语言的场景,例如 DSL(领域特定语言)设计:

infix fun <T> T.shouldEqual(expected: T) {if (this != expected) throw AssertionError("Excepted $expected, but got $this")
}// 使用
actualValue shouldEqual expectedValue
2.2 数学或逻辑表达式

简化数学运算符或逻辑操作的表达:

infix fun Int.pow(exponent: Int): Int = toDouble().pow(exponent).toInt()val result = 2 pow 3 // 等价于 2.pow(3)
2.3 键值对构造(如 to 函数)

Kotlin 标准库中的 to 函数是典型的中缀函数:

val map = mapOf("name" to "Eileen","age" to 30
)
2.4 链式操作

结合中缀函数和扩展函数实现链式调用:

infix fun String.concatWith(another: String) = "$this$another"// 链式中缀调用
val message = "Hello" concatWith "World" concatWith "!"

3 与普通函数相比

中缀函数和普通函数

4 注意事项

  • 参数限制:中缀函数只能有一个参数,且不能有默认值;
  • 可见性:中缀函数必须是 publicinternal(默认,模块内可见);
  • 优先级:中缀函数的优先级是低于算数运算符,但高于布尔运算符;
1 + 2 and 3 // 等价于 (1 + 2) and 3

5 标准库中的经典案例

Kotlin 标准库广泛使用中缀函数。

5.1 to 函数

用于创建 Pair 对象:

infix fun <A, B> A.to(that: B): Pair<A, B> = Pair(this, that)
5.2 until 函数

生成区间:

val range = 1 until 10 // 等价于 1.until(10)
5.3 集合操作

例如 stepdownTo

for (i in 10 downTo 1 step 2) {}

相关文章:

  • C++23 ranges::to:范围转换函数 (P1206R7)
  • iOS视频编码详细步骤(视频编码器,基于 VideoToolbox,支持硬件编码 H264/H265)
  • Linux内核可配置的参数
  • 部署安装gitlab-ce-17.9.7-ce.0.el8.x86_64.rpm
  • QT之信号与槽
  • 深入理解ThingsBoard的Actor模型
  • std::ratio<1,1000> 是什么意思?
  • C语言_自定义类型:结构体
  • JT/T 808 各版本协议字段级别对比与解析适配建议
  • Spring AI 集成 Mistral AI:构建高效多语言对话助手的实战指南
  • 全流量解析:让安全防御从“被动挨打”升级为“主动狩猎”
  • 【Linux网络】网络层
  • RabbitMQ 快速上手:安装配置与 HelloWorld 实践(二)
  • maven项目, idea右上角一直显示gradle的同步标识, 如何去掉
  • Restfull API 风格规则以及特点
  • Unity Image组件无法阻挡手势的解决办法
  • JS逆向实战四:某查查请求头逆向解密
  • 鸿蒙OSUniApp开发富文本编辑器组件#三方框架 #Uniapp
  • STM32F103_LL库+寄存器学习笔记23 - PWM波形输出及软件方式调整周期与占空比
  • 【行为型之访问者模式】游戏开发实战——Unity灵活数据操作与跨系统交互的架构秘诀
  • 人民日报:从“轻微免罚”看涉企执法方式转变
  • 外交部:各方应为俄乌双方恢复直接对话创造条件
  • 重庆市委原常委、政法委原书记陆克华被决定逮捕
  • 广东省中医院脾胃病科大科主任张北平病逝,年仅52岁
  • 李强会见巴西总统卢拉
  • 西北大学副校长范代娣成陕西首富?系家庭财富,本人已从上市公司退出