Kotlin允许我们为自己的类型提供预定义的一组操作符的实现。这些操作符具有固定的符号表示(如 + 或 *) 和固定的优先级。为实现这样的操作符,我们为相应的类型(即二元操作符左侧的类型和一元操作符的参数类 型)提供了一个固定名字的成员函数或扩展函数。重载操作符的函数需要用 operator 修饰符标记。

  另外,我们描述为不同操作符规范操作符重载的约定。

一元操作

  一元前缀操作符

表达式 翻译为
+a a.unaryPlus()
-a a.unaryMinus()
!a a.not()

  这个表是说,当编译器处理例如表达式 +a 时,它执行以下步骤

    — 确定 a 的类型,令其为 T;

    — 为接收者 T 查找一个带有 operator 修饰符的无参函数 unaryPlus(),即成员函数或扩展函数;

    — 如果函数不存在或不明确,则导致编译错误;

    — 如果函数存在且其返回类型为 R,那就表达式 +a 具有类型 R;

  注意 这些操作以及所有其他操作都针对基本类型做了优化,不会为它们引入函数调用的开销。

  以下是如何重载一元减运算符的示例

data class Point(val x: Int, val y: Int)

operator fun Point.unaryMinus() = Point(-x, -y) 

val point = Point(10, 20)

fun main() {
println(-point) // 输出“Point(x=-10, y=-20)”
}

  

递增与递减

表达式 翻译为
a++ a.inc()
a-- a.dec()

  inc() 和 dec() 函数必须返回一个值,它用于赋值给使用 ++ 或 -- 操作的变量。它们不应该改变在其上 调用 inc() 或 dec() 的对象

  编译器执行以下步骤来解析后缀形式的操作符,例如 a++

    — 确定 a 的类型,令其为 T

    — 查找一个适用于类型为 T 的接收者的、带有 operator 修饰符的无参数函数 inc()

    — 检测函数的返回类型是 T 的子类型。

  计算表达式的步骤是

    — 把 a 的初始值存储到临时存储 a0 中

    — 把 a0.inc() 结果赋值给 a

    — 把 a0 作为表达式的结果返回。

  对于 a--,步骤是完全类似的。

  对于前缀形式 ++a 和 --a 以相同方式解析,其步骤是:

    — 把 a.inc() 结果赋值给 a 

    — 把 a 的新值作为表达式结果返回

二元操作

算术运算符

表达式 翻译为
a+b a.plus(b)
a-b a.minus(b)
a * b a.times(b)
a / b a.div(b)
a % b a.rem(b) 、a.mod(b) (已弃用)
a..b a.rangeTo(b)

  对于此表中的操作,编译器只是解析成翻译为列中的表达式。

  请注意,自 Kotlin 1.1 起支持 rem 运算符。Kotlin 1.0 使用 mod 运算符,它在 Kotlin 1.1 中被弃用

示例:下面是一个从给定值起始的 Counter 类的示例,它可以使用重载的 + 运算符来增加计数

data class Counter(val dayIndex: Int) {
operator fun plus(increment: Int): Counter {
return Counter(dayIndex + increment) }
}

“In”操作符

表达式 翻译为
a in b b.contains(a)
a !in b !b.contains(a)

  对于 in 和 !in,过程是相同的,但是参数的顺序是相反的

索引访问操作符

表达式 翻译为
a[i] a.get(i)
a[i,j] a.get(i,j)
a[i_1,......,i_n] a.get(i_1,......,i_n)
a[i] = b a.set(i,b)
a[i,j] = b a.set(i,j,b)
a[i_1 , ...... ,i_n] = b a.set(i_1, ...... , i_n , b)

  

  方括号转换为调用带有适当数量参数的 get 和 set

调用操作符

表达式 翻译为
a() a.invoke()
a(i) a.invoke(i)
a(i , j) a.invoke(i , j)
a(i_1 , ..... , i_n) a.invoke(i_1,.......,i_n)

  圆括号转换为调用带有适当数量参数的 invoke

广义赋值

表达式 翻译为
a += b a.plusAssign(b)
a -= b a.minusAssign(b)
a *= b a.timesAssign(b)
a /= b a.divAssign(b)
a %= b a.remAssign(b) , a.modAssign(b)(已弃用)

  对于赋值操作,例如 a += b,编译器执行以下步骤:

    — 如果右列的函数可用

    — 如果相应的二元函数(即 plusAssign() 对应于 plus())也可用,那么报告错误(模糊),

    — 确保其返回类型是 Unit,否则报告错误,

    — 生成 a.plusAssign(b) 的代码;

    — 否则试着生成 a = a + b 的代码(这里包含类型检测:a + b 的类型必须是 a 的子类型)。

  注意:赋值在 Kotlin 中不是表达式

相等与不等操作符

表达式 翻译为
a == b a?.equals(b) ?: (b === null)
a != b !(a?.equals(b) ? : (b===null))

  这些操作符只使用函数 equals(other: Any?): Boolean,可以覆盖它来提供自定义的相等性检测实现。 不会调用任何其他同名函数(如 equals(other: Foo))。

  注意:=== 和 !==(同一性检测)不可重载,因此不存在对他们的约定。

  这个 == 操作符有些特殊:它被翻译成一个复杂的表达式,用于筛选 null 值。null == null 总是 true,

  对于非空的 x ,x == null 总是 false 而不会调用 x.equals()

比较操作符

表达式 翻译为
a > b a.compareTo(b) > 0
a < b a.compareTo(b) < 0
a >= b a.compareTo(b) >= 0
a <= b a.compareTo(b) <= 0

  所有的比较都转换为对 compareTo 的调用,这个函数需要返回 Int 值

属性委托操作符

  provideDelegate 、getValue 以及 setValue 操作符函数已在委托属性中描述

具名函数的中缀调用

  我们可以通过中缀函数的调用 来模拟自定义中缀操作符

例:

infix fun Person.text(s:Student):Person = Person().plus(s)

  

kotlin更多语言结构——>操作符重载的更多相关文章

  1. lua 14 metatable (类似操作符重载)

    转自:http://www.runoob.com/lua/lua-metatables.html 感性认识: “Lua中Metatable这个概念, 国内将他翻译为元表. 元表为重定义Lua中任意一个 ...

  2. Kotlin操作符重载:把标准操作加入到任何类中(KAD 17)

    作者:Antonio Leiva 时间:Mar 21, 2017 原文链接:https://antonioleiva.com/operator-overload-kotlin/ 就像其他每种语言一样, ...

  3. Kotlin——最详细的操作符与操作符重载详解(上)

    本篇文章为大家详细的介绍Koltin特有的操作符重载.或许对于有编程经验的朋友来说,操作符这个词绝对不陌生,就算没有任何编辑基础的朋友,数学中的算数运算符也绝不陌生.例如(+.-.*./.>.& ...

  4. Kotlin 之操作符重载

    Kotlin 之操作符重载   参考: kotlin in action kotlin 官方参考文档 运算符重载 Kotlin允许我们为自己的类型提供预定义的一组操作符实现(这些操作符都对应的成员函数 ...

  5. kotlin之操作符重载

    一元操作符 表达式 对应的函数 +a a.unaryPlus() -a a.unaryMinus() !a a.not() a++ a.inc() a-- a.dec() fun main(arg: ...

  6. KOTLIN开发语言文档(官方文档) -- 2.基本概念

    网页链接:https://kotlinlang.org/docs/reference/basic-types.html 2.   基本概念 2.1.  基本类型 从可以在任何变量处理调用成员函数和属性 ...

  7. Java 语言结构【转】

    Java 语言结构 基础:包(Package).类(Class)和对象(Object) 了解 Java 的包(Package).类(Class)和对象(Object)这些基础术语是非常重要的,这部分内 ...

  8. 06. Go 语言结构体

    Go语言结构体(struct) Go 语言通过用自定义的方式形成新的类型,结构体是类型中带有成员的复合类型.Go 语言使用结构体和结构体成员来描述真实世界的实体和实体对应的各种属性. Go 语言中的类 ...

  9. Kotlin开发语言文档(官方文档)-- 目录

    开始阅读Kotlin官方文档.先上文档目录.有些内容还未阅读,有些目录标目翻译还需琢磨琢磨.后续再将具体内容的链接逐步加上. 文档链接:https://kotlinlang.org/docs/kotl ...

  10. php入门 数据类型 运算符 语言结构语句 函数 类与面向对象

    php PHP-enabled web pages are treated just like regular HTML pages and you can create and edit them ...

随机推荐

  1. 【Vue】Re20 VueX 第一部分(共享数据,Getters)

    一.概述 专门为VueJS应用程序开发的状态管理模式 集中式存储管理应用的所有组件的状态,按照相应的规则保证状态以一种可预测的方式发生变化 VueX也集成到了官方调试工具devtools extens ...

  2. Inno Setup 出现 the drive or unc share you selected does not exist or is not accessible 解决记录

    背景 软件是使用Inno Setup的,且安装后,再次安装是默认安装到历史路径.一次用户电脑维修后,发现再次安装后报错 解决办法 取消自动安装到默认路径就好了~ UsePreviousAppDir=n ...

  3. C#开发的全屏图片切换效果应用 - 开源研究系列文章 - 个人小作品

    这天无聊,想到上次开发的图片显示软件< PhotoNet看图软件 >,然后想到开发一个全屏图片切换效果的应用,类似于屏幕保护程序,于是就写了此博文.这个应用比较简单,主要是全屏切换换图片效 ...

  4. Apache DolphinScheduler中ZooKeeperCDH不兼容问题的解决方案

    背景 看到Apache DolphinScheduler社区群有很多用户反馈和讨论这块问题,针对不兼容的问题,不仅需要自己重新编译各一个新包,而且因为默认是使用zk-3.8的配置,所以会出现不兼容问题 ...

  5. Sentry 产品指南文档(附:详细脑图整理)

    Sentry 基础知识 https://docs.sentry.io/product/ https://docs.sentry.io/product/sentry-basics/ 问题 https:/ ...

  6. CSP_J2023总结

    维护中 T1 [CSP-J 2023] 小苹果 题目描述 小 Y 的桌子上放着 $n$ 个苹果从左到右排成一列,编号为从 $1$ 到 $n$. 小苞是小 Y 的好朋友,每天她都会从中拿走一些苹果. 每 ...

  7. Win32_GDI_绘制文字路径透明窗口

    效果图: 前面字体是个透明窗口 后面是桌面背景 代码实现: void MyMainDialog::TextPathWindow(LPCTSTR lpShowText) { HDC hdc = GetD ...

  8. 【Python自动化】之利用JS操作页面滚动条(新)

    如何操作页面中的滚动条,有以下几个方法: 一.查找可见元素进行滚动页面(通用) 方法一: ①移动到元素element对象的"顶端",与当前窗口的"顶部"对齐(默 ...

  9. Cannot find loader com.jme3.scene.plugins.ogre.MeshLoader

    五月 20, 2022 2:46:07 下午 com.jme3.asset.AssetConfig loadText 警告: Cannot find loader com.jme3.scene.plu ...

  10. MySQL服务端innodb_buffer_pool_size配置参数

    innodb_buffer_pool_size是什么? innodb_buffer_pool是 InnoDB 缓冲池,是一个内存区域保存缓存的 InnoDB 数据为表.索引和其他辅助缓冲区.innod ...