Kotlin之扩展函数和运算符重载

1、扩展函数

扩展函数就是在不修改源码的情况下,向某个类中新增加函数。下面举个例子:我们想统计一个字符串中字母的数量,我们一般的写法如下:

 fun getLettersCount(str:String):Int{
        var count=0
        for(chr in str){
            if(chr.isLetter())
                count++
        }
        return count
    }

这种写法绝对可以正常工作,但是有了扩展函数之后就不一样了,我们可以使用一种更加面向对象的方式来实现这个功能,先来看下扩展函数的语法结构:

fun ClassName.methodName(params1:Int,params:Int):Int{}

可以看到和普通的方法类似,只不过在方法名之前加上了ClassName.,学习了语法结构后我们来实现一下上面的功能。

  • 1、新建文件String.kt
    文件名不是固定的,只不过我们给String类新增扩展函数,所以命名为String更方便查找和维护。
  • 2、添加顶层方法
fun String.getLettersCount():Int{
    var count=0
    for(chr in this){
        if(chr.isLetter())
            count++
    }
    return count
}

可以看到给String类新增扩展函数,那么函数中就自动拥有了String实例的上下文,因此函数不再接收已String参数了,直接遍历本身即可,this就代表着字符串本身。
注意:扩展函数不一定非要定义成顶层函数,可以定义在任何一个类中。

  • 3、调用
 "Start eating fruit".getLettersCount()

除了String类我们可以向任意类添加扩展函数。

2、运算符重载

Java中有许多内置的运算符关键字:+ - * / % ++ --,在Kotlin中允许对所有的运算符甚至其他的关键字进行重载,在进行运算符 重载后我们可以实现任意对象进行相加或者其他更多的操作。
运算符重载使用的是operator关键字,只要在指定的函数前加上operator关键字就可以实现运算符的重载,指定的函数指的是什么?下面我们看张表

语法糖表达式 指定函数
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++ a.inc()
a-- a.dec()
+a a.unaryPlus()
-a a.unaryMinus()
!a a.not()
a==b a.equals(b)
a>b a.compareTo(b)
a..b a.rangeTo(b)
a[b] a.get(b)
a[b]=c a.set(b,c)
a in b b contains(a)

我们这里以加号运算符为例,来实现两个对象相加的功能。

class Money(val value: Int) {
     //运算符重载
    operator fun plus(money: Money): Money =Money(value+money.value)
}
//调用
 val money:Money=Money(12)+Money(23)

Kotlin允许我们对同一个运算符进行多重重载,下面我们再重载plus函数

class Money(val value: Int) {
     //运算符重载
    operator fun plus(money: Money): Money =Money(value+money.value)
    operator fun plus(newValue:Int):Money=Money(value+newValue)
}
//调用 他们输出结果一致
val money:Money=Money(12)+ Money(23)
val newMoney:Money=Money(12)+23

下面再看一个实例:

fun getRandomLengthString(str:String):String{
        val stringBuilder=StringBuilder()
        val n=Random().nextInt(20)+1
        repeat(n){
            stringBuilder.append(str)
        }
        return stringBuilder.toString()
    }

上面代码的功能就是让字符串重复n遍,那么我们能不能直接使用str*n来实现呢?
由于要实现str*n它对应的指定函数是str.times(n),所以就要把函数定义在String类中,此时我们就可以借助我们刚刚学习的扩展函数,然后结合运算符重载,定义如下:

operator fun String.times(n: Int): String {
    val stringBuilder = StringBuilder()
    repeat(n) {
        stringBuilder.append(this)
    }
    return stringBuilder.toString()
}

由于String内部实现了repeat(n)用于将字符串重复n遍的函数,所以上面代码可简化为:

operator fun String.times(n: Int)=this.repeat(n)

调用如下:

str*(Random().nextInt(20)+1)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352