Kotlin常用操作符

操作符

?操作符

表示这个对象可能为空

1
2
//在变量后面加?,代表这个变量可以为空
var name: String? = "zhangsan"
1
2
3
4
//如果str不能转为Int类型,则返回null
fun parseInt(str: String): Int?{
//
}
1
2
// 如果b非空,就返回b.length,否则返回null,这个表达式的类型是Int?
b?.length

?:操作符

如果操作符左边表达式不为空用左边表达式结果,为空则用右边的值。类似三元运算符

1
2
3
4
val l: Int = if(b!=null) b.length else -1
// 除了完整的if表达式,这个还可以用Elvis操作符来表示:
val l = b?.length ?: -1
// 如果?: 左侧表达式非空,elvis操作符就返回左侧表达式的值,否则返回右侧的值。

注意:当且仅当左侧为空时,才会对右侧表达式求值。

!!操作符

一定是非空的值

1
2
//这会返回一个非空的b值,或者如果b为空,就会抛出空指针异常
var l = b!!.length

== 与 ===

这个类似java中的 == 和 .equals()

== 判断值是否相等,=== 判断值及引用是否完全相等。

1
2
3
4
5
val num: Int = 128;
val a:Int? = num
val b:Int? = num
println(a == b) //true
print(a === b) //false

..符号以及 in 和 !in 操作符

..代表从x到y,==包括x和y,这是一个闭区间==运算符,而until则是半闭区间运算符,代表从a到b范围内所有的值,==包括a和不包括b==。
in代表在一个区间中,!in代表不在一个区间中。

1
2
3
4
5
6
if(i in 1..10){ //等价于 1 <= i <= 10
println(i)
}

使用until函数,创建一个不包括其结束元素的区间
for(i in 1 until 10){} // i in [1,10)

downTo()函数

倒叙遍历,区间内循环

1
for(i in 4 downTo 1){} //print(i) -- 4321

step函数

可以进行任意数量的迭代

1
2
for (i in 1..4 step 2) print(i) // prints "13"
for (i in 4 downTo 1 step 2) print(i) // prints "42"

::符号

得到类的Class对象

1
startActivity(Intent(this@KotlinActivity, MainActivity::class.java))

@符号

限定this的类型

1
2
3
4
5
6
7
8
9
10
11
12
class User {
inner class State{
fun getUser(): User{
//返回User
return this@User
}
fun getState(): State{
//返回State
return this@State
}
}
}

作为标签
跳出双层循环

1
2
3
4
5
6
7
8
9
10
11
12
loop@ for (itemA in arraysA) {
var i : Int = 0
for (itemB in arraysB) {
i++
if (itemB > 2) {
break@loop
}

println("itemB:$itemB")
}

}

==命名函数自定义标签:==

1
2
3
4
5
6
7
8
9
10
11
12
13
14
fun fun_run(){
run {
println("lambda")
}
var i: Int = run {
return@run 1
}
println("$i")
//匿名函数可以通过自定义标签进行跳转和返回
i = run (outer@{
return@outer 2
})
println(i)
}

从forEach跳出

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
fun forEach_label(ints: List<Int>)
{
var i =2
ints.forEach {
//forEach中无法使用continue和break;
// if (it == 0) continue //编译错误
// if (it == 2) /*break //编译错误 */
print(it)
}
run outer@{
ints.forEach {
if (it == 0) return@forEach //相当于在forEach函数中continue,实际上是从匿名函数返回
if (it == 2) return@outer //相当于在forEach函数中使用break,实际上是跳转到outer之外
}
}

if (i == 3)
{
//每个函数的名字代表一个函数地址,所以函数自动成为标签
return@forEach_label //等同于return
}
}

as?操作符

当使用as转型的时候,可能经常出现ClassCastException。所以,现在可以用==as?安全转型==,当转型不成功的时候,它会返回null。

注意:在使用intent传值的时候,会出现空字符串不能用as强制转型,应该使用as?

1
val m: Int? = a as? Int

冒号:

用于类的继承,变了的定义

  1. 类型和超类型之间的冒号前要有一个空格
  2. 实例和类型之间的冒号前不能用空格
1
2
3
4
5
//定义全局变量时
var str: String? = null

//类的继承与变量定义
class TestActivity<T : Serializable>(str: String) : Activity{}

类型判断符is

检查某个实例是否是某个类型,如果判断出属于某个类型,那么判断后的分支中可以直接可当该类型使用,==无需显示转换==

1
2
3
4
5
6
fun getStringLength(obj: Any): Int? {
//obj在&&右边自动动转换成"String"类型
if (obj is String && obj.length > 0)
return obj.length
return null
}

$操作符

字符串可以包含模板表达式,及一小段代码,会求值并把结果包含到字符串中。模板字符串以美元符号$开头,由一个简单的名字构成:

1
2
val value:Int=5;
val str:String="the value is $value"

标准函数

run 、 apply 、 let 、 also 和 with 五个函数均位于 kotlin 包下的 Standard 文件中,其含义和用法比较相似,现分别介绍如下。

run

  • run函数使用的一般结构
1
2
object.run{
}
  • run函数的inline+lambda结构
1
2
@kotlin.internal.InlineOnly
public inline fun <T, R> T.run(block: T.() -> R): R = block()
  • 结构分析
    run函数实际上可以说是let和with的结合体,run函数只是接受一个lambda函数作为参数,以闭包形式返回,==返回值为最后一行的值或者指定的return的表达式==

  • 例子

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
//kotlin

fun main(args: Array<String>) {
val user = User("Kotlin", 1, "1111111")

val result = user.run {
println("my name is $name, I am $age years old, my phone number is $phoneNum")
1000
}
println("result: $result")
}

//java

public static final void main(@NotNull String[] args) {
Intrinsics.checkParameterIsNotNull(args, "args");
User user = new User("Kotlin", 1, "1111111");
String var5 = "my name is " + user.getName() + ", I am " + user.getAge() + " years old, my phone number is " + user.getPhoneNum();
System.out.println(var5);
int result = 1000;
String var3 = "result: " + result;
System.out.println(var3);
}
  • 适用场景
    适用于let,with函数任何场景,因为run函数是let,with两个函数的结合体,它既弥补了let函数在函数体内必须使用it参数替代对象,在run函数中可以像with函数一样省略,直接访问实例的公有属性和方法,另一方面也弥补了with函数传入对象判空问题,在run函数中可以像let函数一样做判空处理

  • run函数使用前后对比

不使用run函数

1
2
3
4
5
6
7
8
9
10
11
12
13
override fun onBindViewHolder(holder: ViewHolder, position: Int){
val item = getItem(position)?: return

with(item){

holder.tvNewsTitle.text = StringUtils.trimToEmpty(titleEn)
holder.tvNewsSummary.text = StringUtils.trimToEmpty(summary)
holder.tvExtraInf = "难度:$gradeInfo | 单词数:$length | 读后感: $numReviews"
...

}

}

使用run函数

1
2
3
4
5
6
7
8
9
10
11
12
override fun onBindViewHolder(holder: ViewHolder, position: Int){

getItem(position)?.run{
holder.tvNewsTitle.text = StringUtils.trimToEmpty(titleEn)
holder.tvNewsSummary.text = StringUtils.trimToEmpty(summary)
// 可以看到直接省略了item,并且可以在之前判空
holder.tvExtraInf = "难度:$gradeInfo | 单词数:$length | 读后感: $numReviews"
...

}

}

let

  • let函数的一般结构
1
2
3
4
5
6
7
8
9
object.let{
it.todo()//在函数体内使用it替代object对象去访问其公有的属性和方法
...
}

//另一种用途 判断object为null的操作
object?.let{//表示object不为null的条件下,才会去执行let函数体
it.todo()
}
  • let函数底层的inline扩展函数+lambda结构
1
2
@kotlin.internal.InlineOnly
public inline fun <T, R> T.let(block: (T) -> R): R = block(this)
  • 结构分析
    从源码let函数来看,它是一个只有一个lambda 函数块block作为参数的函数,调用T类型对象的let函数,则该对象为函数的参数。在函数块内可以通过it指代该对象,返回值为函数块的最后一行或者指定return表达式。

  • let函数的kotlin和java转化

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
 //kotlin

fun main(args: Array<String>) {
val result = "testLet".let {
println(it.length)
1000
}
println(result)
}
//返回值:7/1000

//java

public final class LetFunctionKt {
public static final void main(@NotNull String[] args) {
Intrinsics.checkParameterIsNotNull(args, "args");
String var2 = "testLet";
int var4 = var2.length();
System.out.println(var4);
int result = 1000;
System.out.println(result);
}
}
  • let函数适用的场景
    最常用的场景就是使用let函数处理需要针对一个可以null的对象统一判空处理。

  • let函数的前后对比

没有使用前:

1
2
3
mVideoPlayer?.setVideoView(activity.course_video_view)
mVideoPlayer?.setControllerView(activity.course_video_controller_view)
mVideoPlayer?.setCurtainView(activity.course_video_curtain_view)

使用后:

1
2
3
4
5
6
//对对象做了统一的判空处理
mVideoPlayer?.let {
it.setVideoView(activity.course_video_view)
it.setControllerView(activity.course_video_controller_view)
it.setCurtainView(activity.course_video_curtain_view)
}

with

  • with函数的一般结构
1
2
with(object){
}
  • with函数底层的inline扩展函数+lambda结构
1
2
@kotlin.internal.InlineOnly
public inline fun <T, R> with(receiver: T, block: T.() -> R): R = receiver.block()
  • 结构分析
    with函数和前面几个函数使用方法略有不同,因为它不是以扩展的形式存在的。它是将某个对象作为函数的参数,==在函数块内可以通过this指代该对象==。返回值为函数块最后一行或者指定的return表达式。

    可以看出with函数接收了两个参数,分别为T类型的对象receiver和一个lambda函数块,所以with函数最原始的样子如下:

1
2
3
4
val result = with(user, {
println("my name is $name, I am $age years old, my phone number is $phoneNum")
1000
})

但是由于,with函数最后一个参数是一个函数,可以把函数提到圆括号外部,所以最终with函数的调用形式如下:(可以看下kotlin的lambda表达式)

1
2
3
4
val result = with(user) {
println("my name is $name, I am $age years old, my phone number is $phoneNum")
1000
}
  • with函数的kotlin和java转化
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
//kotlin

fun main(args: Array<String>) {
val user = User("Kotlin", 1, "1111111")

val result = with(user) {
println("my name is $name, I am $age years old, my phone number is $phoneNum")
1000
}
println("result: $result")
}

//java

public static final void main(@NotNull String[] args) {
Intrinsics.checkParameterIsNotNull(args, "args");
User user = new User("Kotlin", 1, "1111111");
String var4 = "my name is " + user.getName() + ", I am " + user.getAge() + " years old, my phone number is " + user.getPhoneNum();
System.out.println(var4);
int result = 1000;
String var3 = "result: " + result;
System.out.println(var3);
}
  • with函数的适用场景

适用于调用同一个类的多个方法时,可以省去类名重复,直接调用类的方法即可,经常用于Android中RecyclerView中onBindViewHolder中,数据model的属性映射到UI上

  • with函数使用前后对比

没有使用:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
@Override
public void onBindViewHolder(ViewHolder holder, int position) {

ArticleSnippet item = getItem(position);
if (item == null) {
return;
}
holder.tvNewsTitle.setText(StringUtils.trimToEmpty(item.titleEn));
holder.tvNewsSummary.setText(StringUtils.trimToEmpty(item.summary));
String gradeInfo = "难度:" + item.gradeInfo;
String wordCount = "单词数:" + item.length;
String reviewNum = "读后感:" + item.numReviews;
String extraInfo = gradeInfo + " | " + wordCount + " | " + reviewNum;
holder.tvExtraInfo.setText(extraInfo);
...
}

使用了之后:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
override fun onBindViewHolder(holder: ViewHolder, position: Int){
val item = getItem(position)?: return

with(item){

holder.tvNewsTitle.text = StringUtils.trimToEmpty(titleEn)
holder.tvNewsSummary.text = StringUtils.trimToEmpty(summary)
//省略了item,直接调用了实例中的方法
holder.tvExtraInf.text = "难度:$gradeInfo | 单词数:$length | 读后感: $numReviews"
...

}

}

apply

  • apply函数使用的一般结构
1
2
3
object.apply{
//todo
}
  • apply函数的inline+lambda结构
1
2
@kotlin.internal.InlineOnly
public inline fun <T> T.apply(block: T.() -> Unit): T { block(); return this }
  • apply函数的inline结构分析

从结构上来看,apply函数和run函数很像,唯一不同点就是他们各自返回值不同,run函数是以闭包形式返回最后一行代码的值,而apply函数的返回值则是传入对象的本身。

  • apply函数的kotlin和java转化
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
/kotlin

fun main(args: Array<String>) {
val user = User("Kotlin", 1, "1111111")

val result = user.apply {
println("my name is $name, I am $age years old, my phone number is $phoneNum")
1000
}
println("result: $result")
}
//输出:result:kotlin,1,111111(User本身对象被打印出来)


//java

public final class ApplyFunctionKt {
public static final void main(@NotNull String[] args) {
Intrinsics.checkParameterIsNotNull(args, "args");
User user = new User("Kotlin", 1, "1111111");
String var5 = "my name is " + user.getName() + ", I am " + user.getAge() + " years old, my phone number is " + user.getPhoneNum();
System.out.println(var5);
String var3 = "result: " + user;
System.out.println(var3);
}
  • apply函数使用前后的对比

没有使用:

1
2
3
4
5
6
7
mSheetDialogView = View.inflate(activity, R.layout.biz_exam_plan_layout_sheet_inner, null)
mSheetDialogView.course_comment_tv_label.paint.isFakeBoldText = true
mSheetDialogView.course_comment_tv_score.paint.isFakeBoldText = true
mSheetDialogView.course_comment_tv_cancel.paint.isFakeBoldText = true
mSheetDialogView.course_comment_tv_confirm.paint.isFakeBoldText = true
mSheetDialogView.course_comment_seek_bar.max = 10
mSheetDialogView.course_comment_seek_bar.progress = 0

使用后:

1
2
3
4
5
6
7
8
9
10
11
//形成了链式调用
mSheetDialogView = View.inflate(activity, R.layout.biz_exam_plan_layout_sheet_inner, null).apply{

course_comment_tv_label.paint.isFakeBoldText = true
course_comment_tv_score.paint.isFakeBoldText = true
course_comment_tv_cancel.paint.isFakeBoldText = true
course_comment_tv_confirm.paint.isFakeBoldText = true
course_comment_seek_bar.max = 10
course_comment_seek_bar.progress = 0

}

多层级判空问题

1
2
3
4
5
6
7
8
9
10
11
12
13
if (mSectionMetaData == null || mSectionMetaData.questionnaire == null || mSectionMetaData.section == null) {
return;
}

if (mSectionMetaData.questionnaire.userProject != null) {
renderAnalysis();
return;
}

if (mSectionMetaData.section != null && !mSectionMetaData.section.sectionArticles.isEmpty()) {
fetchQuestionData();
return;
}

-> 使用了apply后可以变为链式调用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
mSectionMetaData?.apply{

//mSectionMetaData不为空的时候操作mSectionMetaData

}?.questionnaire?.apply{

//questionnaire不为空的时候操作questionnaire

}?.section?.apply{

//section不为空的时候操作section

}?.sectionArticle?.apply{

//sectionArticle不为空的时候操作sectionArticle

}

also

  • also函数使用的一般结构
1
2
3
object.also{
//todo
}
  • also函数的inline+lambda结构
1
2
3
4
@kotlin.internal.InlineOnly
@SinceKotlin("1.1")
//可以看到这个函数是从kotlin1.1开始才有的
public inline fun <T> T.also(block: (T) -> Unit): T { block(this); return this }
  • also函数的inline结构分析

also函数的结构实际上和let很像,唯一的区别是返回值不一样,let是以闭包的形式返回的,返回函数体内最后一行的值,如果最后一行为空就返回一个Unit类型的默认值,而also函数则返回传入对象本身。这个和apply一样。

  • also函数的kotlin和java转化
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
//kotlin

fun main(args: Array<String>) {
val result = "testLet".also {
println(it.length)
1000
}
println(result)
}
//结果:7/testLet

//java

public final class AlsoFunctionKt {
public static final void main(@NotNull String[] args) {
Intrinsics.checkParameterIsNotNull(args, "args");
String var2 = "testLet";
int var4 = var2.length();
System.out.println(var4);
System.out.println(var2);
}
}

标准函数总结

函数名 定义inline结构 函数体内代指对象 返回值 是否是扩展函数 使用场景
let fun T.let(block: (T) -> R): R = block(this) it 闭包返回 适用于处理不为null的操作场景
with fun with(receiver: T, block: T.() -> R): R = receiver.block() this 返回闭包 适用于调用同一个类的多个方法时,可以省去类名重复,直接调用类的方法即可,经常用于Android中RecyclerView中onBinderViewHolder中,数据model的属性映射到UI上
run fun T.run(block: T.() -> R): R = block() this 闭包返回 适用于let,with函数任何场景。
apply fun T.apply(block: T.() -> Unit): T { block(); return this } this 返回this
also fun T.also(block: (T) -> Unit): T { block(this); return this } it 返回this 适用于let函数的任何场景,一般可用于多个扩展函数链式调用