Malinkang's blog

人生最可悲的事情,莫过于胸怀大志,却又虚度光阴。

你好,我是马林康,一名Android开发者。


《Kotlin实战》读书笔记 第7章 运算符重载及其他约定

7.1 重载算术运算符

7.1.1 重载二元算术运算

data class Point(val x: Int, val y: Int) {
operator fun plus(other: Point): Point {
return Point(x + other.x, y + other.y)
}
}
val p1 = Point(10, 20)
val p2 = Point(30, 40)
println(p1 + p2) //Point(x=40, y=60)
operator fun Point.plus(other: Point): Point {
return Point(x + other.x, y + other.y)
}
表达式 函数名
a * b times
a / b div
a % b mod
a + b plus
a - b minus
operator fun Point.times(scale: Double): Point {
return Point((x * scale).toInt(), (y * scale).toInt())
}
val p = Point(10, 20)
print(p * 1.5) //Point(x=15, y=30)

Kotlin运算符不会自动支持交换性。如果希望用户能够使用15*p以外,还能使用p*1.6,你需要为它定义个单独的运算符:

operator fun Double.times(p:Point):Point{
return Point((p.x * this).toInt(), (p.y * this).toInt())
}
operator fun Char.times(count: Int): String {
return toString().repeat(count)
}
println('a' * 3) //aaa

7.1.2 重载符合运算符

7.1.3 重载一元运算符

operator fun Point.unaryMinus():Point{
return Point(-x,-y)
}
val p = Point(10,20)
println(-p)
表达式 函数名
+a unaryPlus
-a unaryMinus
!a not
++a,a++ inc
–a,a– dec
operator fun BigDecimal.inc() = this +BigDecimal.ONE
var bd = BigDecimal.ZERO
println(bd++) //0
println(++bd) //2

7.2 重载比较运算符

7.2.1 等号运算符:“equals”

data class Point(val x: Int, val y: Int) {
override fun equals(other: Any?): Boolean {
//这里使用了恒等运算符===来检查参数与调用equals的对象是否相同。恒等运算符与Java中的==运算符
//是完全相同的;===运算符不能被重载
if (other === this) return true
if (other !is Point) return false
return other.x == x && other.y == y
}
}
println(Point(10,20)==Point(10,20)) //true
println(Point(10,20)!=Point(5,5)) //true
println(null == Point(1,2)) //false

7.2.2 排序运算符:compareTo

class Person(val firstName: String, val lastName: String) : Comparable<Person> {
override fun compareTo(other: Person): Int {
return compareValuesBy(this, other, Person::lastName, Person::firstName)
}
}

7.3 集合与区间的约定

7.3.1 通过下标来访问元素:“get”和“set”

{% code-tabs %} {% code-tabs-item title="实现get约定" %}
operator fun Point.get(index: Int): Int {
return when (index) {
0 -> x
1 -> y
else ->
throw IndexOutOfBoundsException("Invalid coordinate $index")
}
}
val p = Point(10,20)
println(p[1]) //20
operator fun MutablePoint.set(index: Int, value: Int) {
when (index) {
0 -> x = value
1 -> y = value
else ->
throw IndexOutOfBoundsException("Invalid coordinate $index")

}
}
val p = MutablePoint(10, 20)
p[1] = 42
println(p) //MutablePoint(x=10, y=42)

7.3.2 “in”的约定

in运算符,用于检查某个对象是否属于集合。相应的函数叫做contains

data class Rectangle(val upperLeft:Point,val lowerRight:Point)

operator fun Rectangle.contains(p:Point):Boolean{
return p.x in upperLeft.x until lowerRight.x &&
p.y in upperLeft.y until lowerRight.y
}
val rect = Rectangle(Point(10,20),Point(50,50))
println(Point(20,30) in rect) //true
println(Point(5,5) in rect) //false

7.3.3 rangeTo的约定

val now = LocalDate.now()
val vacation = now..now.plusDays(10)
println(now.plusWeeks(1) in vacation) //true
val n = 9
println(0..(n+1)) //0..10
(0..n).forEach(::print) //0123456789

7.3.4 在“for”循环中使用“iterator”的约定

operator fun ClosedRange<LocalDate>.iterator(): Iterator<LocalDate> =
object : Iterator<LocalDate> {
var current = start
override fun hasNext() = current <= endInclusive
override fun next() = current.apply {
current = plusDays(1)
}
}
val newYear = LocalDate.ofYearDay(2017, 1)
val daysOff = newYear.minusDays(1)..newYear
for (dayOff in daysOff) {
println(dayOff)
}
// 2016-12-31
// 2017-01-01
最近的文章

《Kotlin实战》读书笔记 第8章 Lambda作为形参和返回值

8.1 声明高阶函数高阶函数就是以另一个函数作为参数或者返回值的函数。 8.1.1 函数类型 val sum = { x: Int, y: Int -&gt; x + y } val action = { println(42)} run { println( …

于  , 继续阅读
更早的文章

《Kotlin实战》读书笔记 第6章 Kotlin的类型系统

6.1 可空性6.1.1 可空类型 //增加了null检查后,这段代码就可以编译了 fun strLenSafe(s: String?) = if (s != null) s.length else 0 6.1.2 类型的含义6.1.3 安全调用运算符安全调用运算符?允许把一次null检查和一次方法 …

于  , 继续阅读
comments powered by Disqus