Kotlin

news2025/1/10 18:16:13

 

目录

 一、Kotlin 基础语法

1、方法函数

2、常量 val 和变量 var 

 3、${} 字符串模板

 4、null 处理   !!. 不能为空  ?.为空不处理    ?:为空处理成  

5、is 类型转换 相当于  instanceof

 6、Any  相当于 Java的 Object

二、Kotlin 基本数据类型

1、基本数据类型(6种,比java少了boolean/char)

2、数据类型转换

 3、字符  `c`

 4、数组arrayOf、arrayListOf

 5、字符串换行 """xxx"""

 三、Kotlin 条件控制

1、if语句

2、判断是否在区间内 in

 3、When 表达式(相当于 Java switch语句)

四、Kotlin 循环控制

五、Kotlin 类和对象

1、类

2、属性 

 3、自定义set/get方法   field 表示当前属性

 4、 lateinit  延迟初始化关键字

 5、抽象类

 6、嵌套类

7、内部类  inner (可使用外部类属性和方法) 

 8、类的修饰符

 六、Kotlin 继承

1、可以被继承的类都要用 open修饰符

 2、构造函数

3、重写 override 

七、Kotlin 接口

八、Kotlin 扩展

1、扩展函数

 2、扩展一个空对象

3、扩展函数是静态解析的(注意)

 4、拓展属性

5、伴生对象 companion object 相当于Java 静态(修饰符单例)

 九、Kotlin 数据类与密封类

1、数据类 data

十、Kotlin 泛型

1、泛型类

 2、泛型函数

 十一、Kotlin 对象表达式和对象声明 object 

1、对象表达式

 2、对象声明(获取单例)

 十二、kotlin 委托  by

1、委托类

 2、属性委托

 3、懒加载 by lazy{ }

4、可观察属性 Observable 

 5、map/mutableMap

 十三、kotlin 回调 block:

1、block: () -> T    block: (T) -> T   block: T.() -> T

2、Block语法格式

十四、Kotlin中let、run、with、apply及also的差别

1、let  public inline fun  T.let(block: (T) -> R): R ,>

 2、run “this”作为上下文对象,且它的调用方式与let一致。

3、with  public inline fun  with(receiver: T, block: T.() -> R): R ,>

 4、apply public inline fun  T.apply(block: T.() -> Unit): T

5、also public inline fun  T.also(block: (T) -> Unit): T 


 一、Kotlin 基础语法

1、方法函数

    /**
     * fun : 方法关键字
     * sum :方法名
     * a:Int : 传参及参数类型
     * b:Int : 传参及参数类型
     * :Int : 返回值
     */
    fun sum(a:Int,b:Int) :Int{
        return a+b
    }

2、常量 val 和变量 var 

 

 3、${} 字符串模板

示例

object Main {

    private val name :String = "小明"

    @JvmStatic
    fun main(args: Array<String>) {
        val newStr = "这是小明"
        val newStr2 = "这是${name}"
        println(newStr)
        print(newStr2)
    }
}

打印日志如下:

 4、null 处理   !!. 不能为空  ?.为空不处理    ?:为空处理成  

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val age1: String? = null// ? 表示可为空
        val age2: String? = "23"
        // !!. 不能为空 
        val age3 = age1!!.toInt()//做了非空判断,传入null则抛出null异常
        // ?.为空不处理
        val age4 = age1?.toInt() // 相当于   age1?.toInt()== null ->  age4 = age1?.toInt() -> val age4 : String? = null
        // ?:为空处理成  
        val age5 = age1?.toInt() ?: "666"  // 相当  age1?.toInt()== null -> if(null ==age1?.toInt() ) return "666"
    }
}

5、is 类型转换 相当于  instanceof

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        changeType("100")
        changeType(99)
    }

    private fun changeType(obj: Any) {
        if (obj is String) {
            println("${obj}长度是${obj.length}")
        } else {
            println("非 String 类型")
        }
    }
}

日志如下:

 6、Any  相当于 Java的 Object

二、Kotlin 基本数据类型

1、基本数据类型(6种,比java少了boolean/char)

 对比Java

​Java基本类型共有八种,基本类型可以分为三类,字符类型char,布尔类型boolean以及数值类型byte、short、int、long、float、double。数值类型又可以分为整数类型byte、short、int、long和浮点数类型float、double

2、数据类型转换

 3、字符  `c`

示例

 4、数组arrayOf、arrayListOf

示例


object Main {

    @JvmStatic
    fun main(args: Array<String>) {

        val array = arrayOf(1, 2, 3)

        val arrayList = arrayListOf<Person>()
        arrayList.add(Person("xiaoming"))
        arrayList.add(Person("xiaohong"))

        for (a in array) {
            println(a.toString())
        }
        for (p: Person in arrayList) {
            println(p.name)
        }
    }

    class Person(val name: String) {

    }
}

日志如下:

 5、字符串换行 """xxx"""

以为kotlin不支持加号字符串换行

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val str = "hello " +
                "world " +
                "!"
        println(str)
    }
}

日志如下:

 所以需要"""xxx"""来实现字符串换行显示

示例:  trimIndent() 去除前置空格

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val str = """
            Hellow
            World
            !
        """.trimIndent()
        println(str)
    }
}

日志如下:

 三、Kotlin 条件控制

1、if语句

kotlin没有Java的三元运算

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val a: Int = 100
        val b: Int = 99
        val max: Int = if (a > b) a else b

        val max2: Int = if (a < b) {
            println("a<b")//先执行
            a
        } else {
            println("a>b")//先执行
            b
        }
        println(max)//后执行
        println(max2)//后执行
    }
}

日志如下

2、判断是否在区间内 in

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val a = 2
        val b = 10

        if (a in 5..10) {
            println("a 在 5到10 区间")
        } else {
            println("a 不在 5到10 区间")
        }
        if (b in 5..10) {
            println("b 在 5到10 区间")
        } else {
            println("b 不在 5到10 区间")
        }
    }
}

日志如下

 3、When 表达式(相当于 Java switch语句)

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val a = 3

        when(a){
            1 -> print("1")
            2 -> print("2")
            else -> print("0")
        }
    }
}

如果很多分支需要用相同的方式处理,则可以把多个分支条件放在一起,用逗号分隔:

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val a = 3

        when(a){
            1,2,3,4 -> print("1")
            5 -> print("2")
            else -> print("0")
        }
    }
}

四、Kotlin 循环控制

1、for循环 ,withIndex()  indices

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val arrayList = arrayListOf<String>("1", "2", "3")
        for (a in arrayList) {
            println(a)
        }
        //withIndex() 带位置标示
        for ((index, value) in arrayList.withIndex()) {
            println("index $index value is $value")
        }
        //indices
        for (index  in arrayList.indices) {
            println("index $index value is ${arrayList[index]}")
        }
    }
}

日志如下

五、Kotlin 类和对象

1、类

2、属性 

 3、自定义set/get方法   field 表示当前属性

有时候会对后台返回的对象属性做键值转换,例如 性别sex返回 1:男 2:女,希望getSex时直接返回其对应值 

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {

        val person = Person()
        person.name = "xiao ming"
        person.age = 18
        person.sex = "1"
        println(person.name)
        println(person.age)
        println(person.sex)
    }
}

class Person() {
    //field 表示当前属性

    var name: String? = null
        get() = field?.toUpperCase()

    var age: Int = 0
        set(value) {
            if (value >= 18) {
                field = value - 10
            } else {
                field = value + 10
            }
        }

    var sex: String? = null
        get() = if ("1" == field) {
            "男"
        } else {
            "女"
        }
}

日志如下

 4、 lateinit  延迟初始化关键字

 非空属性必须在定义的时候初始化

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
    }

    class Person(val name:String){
        lateinit var bean:String //定义变量不初始化会报错时 则可以使用 lateinit  确定后面一定会初始化
    }
}

 5、抽象类

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val animalA = Cat()
        val animalB = Dog()
        animalA.say()
        animalB.say()
    }
}

class Cat : Animal() {
    override fun say() {
        println("喵~")
    }
}

class Dog : Animal() {
    override fun say() {
        println("汪~")
    }
}

//抽象类
abstract class Animal() {
    abstract fun say();//抽象方法
}

日志如下

 6、嵌套类

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val man = Person.Man()
        man.name = "xiao ming"
        println(man.name)
    }
}

class Person(){
    //嵌套类
    class Man(){
        var name: String? = null
    }
}

日志如下

7、内部类  inner (可使用外部类属性和方法) 

 示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val outer = Outer()
        val inner = outer.Inner()
        val demo = inner.foo()
        println(demo) //   1
        val demo2 = inner.innerTest()
        println("被修改的outer.v  = ${outer.v}")   // 内部类可以引用外部类的成员,例如:成员属性
    }
}

class Outer {
    private val bar: Int = 1
    var v = "成员属性"
    /**嵌套内部类**/
    inner class Inner {
        fun foo() = bar  // 访问外部类成员
        fun innerTest() {
            val o = this@Outer //获取外部类的成员变量
            println("内部类可以引用外部类的成员,例如:" + o.v)
            o.v = "111"
            println("内部类可以修改引用外部类的成员,例如:" + o.v)
        }
    }
}

日志如下

 8、类的修饰符

 六、Kotlin 继承

1、可以被继承的类都要用 open修饰符

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val p = Person("xiao ming")
        println(p.name)
        println(p.str)
    }
}

class Person(val name: String) : Base("child is Person, child name is $name") {
}

//open 关键词 允许当前类被继承
open class Base(val str: String) {
}

日志如下

 2、构造函数

 1)子类有构造函数,则要在子类构造初始化父类构造

 示例

class Person(name: String, age: Int, sex: String) : Base(name) {
}

//open 关键词 允许当前类被继承
open class Base(name: String) {
}

 2)子类无构造函数,则需要使用 constructor():super()

 示例

class Person : Base {
    constructor():super("person")
}

//open 关键词 允许当前类被继承
open class Base(val name: String) {
}

3、重写 override 

1)在基类中,使用fun声明函数时,此函数默认为final修饰,不能被子类重写。如果允许子类重写该函数,那么就要手动添加 open 修饰它, 子类重写方法使用 override 关键词:

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val p = Person()
        p.say("xiao ming")
    }
}

class Person : Base("Person") {

    override fun say(name: String) {// 重写方法
        println("person $name")
    }
}

//open 关键词 允许当前类被继承/重写
open class Base(val name: String) {

    open fun say(name: String) {// 允许重写方法
        println("base name")
    }
}

日志如下

 2)super<A>.f()  如果有多个相同的方法(继承或者实现自其他类,如A、B类),则必须要重写该方法,使用super范型去选择性地调用父类的实现。

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val c = C()
        c.f()
    }
}

open class A {
    open fun f() { println("A") }
    fun a() { println("a") }
}
interface B{
    fun f(){ println("B")}
    fun b(){ println("b") }
}
class C() :A(),B{
    override fun f() {
        super<A>.f()//调用 A.f()
        super<B>.f()//调用 B.f()
    }
}

日志如下

七、Kotlin 接口

1) 接口方法可以提前实现

2) 接口属性被继承时必须赋值

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val p = Person()
        println(p.name)
        p.bar()
        p.foo()
    }
}

class Person : PersonInterface {
    override var name: String = "xiao ming"

    override fun bar() {
        println("bar")
    }
}

interface PersonInterface {
    var name: String//接口属性,接口属性不赋值,继承接口必须赋值

    fun bar()
    fun foo() {
        println("foo")//接口可以提前实现方法体
    }
}

 日志如下

八、Kotlin 扩展

Kotlin 可以对一个类的属性和方法进行扩展,且不需要继承或使用 Decorator 模式。

扩展是一种静态行为,对被扩展的类代码本身不会造成任何影响。

1、扩展函数

示例

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        Person("xiao ming").say()//拓展函数调用
    }
}

//拓展Person类,提供 say()函数
fun Person.say() {
    println("我是拓展函数:$name")
}

class Person(val name: String) {}

日志如下 

 2、扩展一个空对象

在扩展函数内, 可以通过 this 来判断接收者是否为 NULL,这样,即使接收者为 NULL,也可以调用扩展函数。例如:

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val t :Any? = null
        println(t.toString())
    }
}

fun Any?.toString(): String {
    if (null == this) {
        return "null"
    }
    // 空检测之后,“this”会自动转换为非空类型,所以下面的 toString()
    return toString()
}

 日志如下

3、扩展函数是静态解析的(注意)

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        printFoo(D())
    }
}

open class C
class D : C()

fun C.foo() = "c"
fun D.foo() = "d"

fun printFoo(c: C) {
    println(c.foo())//拓展函数是静态,所以不管传值什么,都会调用 fun C.foo() = "c"
}

日志如下: 

 若扩展函数和成员函数一致,则使用该函数时,会优先使用成员函数。

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        D().foo()
    }
}

class D{
    fun foo(){ println("成员函数") }
}

fun D.foo(){ println("拓展函数") }

日志如下:

 4、拓展属性

扩展属性允许定义在类或者kotlin文件中,不允许定义在函数中。初始化属性因为属性没有后端字段(backing field),所以不允许被初始化,只能由显式提供的 getter/setter 定义。 

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val d = D()
        d.name = "name"
        println(d.name)
        println(d.name2)
    }
}

class D {
    var name2: String? = null
}

//拓展属性
var D.name: String
    get() {
        return "xiao ming"
    }
    set(value) {
        println("拓展属性")
        this.name2 = "name2 :$value"
    }

日志如下:

5、伴生对象 companion object 相当于Java 静态(修饰符单例)

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        D.foo()//拓展函数
        println(D.name)//拓展属性

        //companion object 演示
        D.A.age
        D.A.say()

        D.age//静态属性,可以直接调用
        D.say()//静态方法,可以直接调用
    }
}

class D {
    object A {
        val age = "10"
        fun say() {
            println("age:10")
        }
    }

    companion object {
        val age = "20"
        fun say() {
            println("age:20")
        }
    }
}

fun D.Companion.foo() {
    println("伴生对象的扩展函数")
}

val D.Companion.name: String
    get() = "伴生对象的扩展属性"

日志如下:

 九、Kotlin 数据类与密封类

1、数据类 data

 copy()函数使用示例(复制数据的同时,可以修改数据内容)

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val user1 = User("xiaoming", 18)
        val user2 = user1.copy(age = 20)
        
        println("user1: name=${user1.name},age=${user1.age}")
        println("user2: name=${user2.name},age=${user2.age}")
    }
}

data class User(val name: String, val age: Int)

日志如下:

十、Kotlin 泛型

泛型,即 "参数化类型",将类型参数化,可以用在类,接口,方法上。

与 Java 一样,Kotlin 也提供泛型,为类型安全提供保证,消除类型强转的烦恼。

1、泛型类

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val box01 = Box<String>("xiao ming")
        val box02 = Box<Int>(20)
        println(box01.value)
        println(box02.value)
    }
}

//泛型类
class Box<T>(t: T) {
    val value = t
}

日志如下:

 2、泛型函数

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        doSomething(100)
        doSomething("xiao ming")
        doSomething(true)
        doSomething(15.55)
    }

    //泛型函数
    fun <T> doSomething(t: T) {
        when (t) {
            is Int -> println("泛型是Int型")
            is String -> println("泛型是String型")
            is Boolean -> println("泛型是Boolean型")
            else -> println("未判断的类型")
        }
    }
}

 日志如下:

 十一、Kotlin 对象表达式和对象声明 object 

1、对象表达式

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        //object: 对象表达式
        val listener = object : Listener{
            override fun onClick() {
                println("onClick")
            }
        }
        listener.onClick()
    }
}

interface Listener{
    fun onClick()
}

日志如下:

 2、对象声明(获取单例)

Kotlin 使用 object 关键字来声明一个对象。

Kotlin 中我们可以方便的通过对象声明来获得一个单例。

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val user01 = User
        val user02 = User
        user01.name = "xiao ming"// object对象是单例,所以 user02 值也会改变
        println(user01.name)
        println(user02.name)
    }
}

object User{
    var name = ""
    var age = 20
}

 日志如下:

 

 十二、kotlin 委托  by

1、委托类

类委托的核心思想在于将一个类的具体实现委托给另一个类去完成。

意义在于让大部分的方法实现调用辅助对象中的方法,少部分的方法实现由自己来重写,甚至加入一些自己独有的方法。

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val a = Dog("wangwang")
        BlackDog(a).print()
        BlackDog(a).dogPrint()
    }
}

interface Animal {
    fun print()
}

class Dog(val name: String) : Animal {
    override fun print() {
        println("name:${name}")
    }
}

//委托类 BlackDog 委托给 Animal
class BlackDog(a: Animal) : Animal by a {
    //实现自己的方法
    fun dogPrint(){
        println("dogPrint")
    }
}

日志如下 

 2、属性委托

委托属性的核心思想是将一个属性(字段)的具体实现委托给另一个类去完成。

import kotlin.reflect.KProperty

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val dog = Dog()
        dog.name = "dog"
        println(dog.name)
    }
}

class Dog{
    //委托属性
    var name: String by Animal()
}

class Animal{
    operator fun getValue(dog: Dog, property: KProperty<*>): String {
        return "委托属性${property}为wangwang"
    }

    operator fun setValue(dog: Dog, property: KProperty<*>, s: String) {
        println("property:${property}, value:${s}")
    }
}

 日志如下:

 3、懒加载 by lazy{ }

by lazy并不是连在一起的关键字,只有by才是Kotlin中的关键字,lazy在这里只是一个高阶函数而已。

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        println(lazyValue)   // 第一次执行,执行两次输出表达式
        println(lazyValue)   // 第二次执行,只输出返回值
    }

    val lazyValue: String by lazy {
        println("computed!")     // 第一次调用输出,第二次调用不执行
        "Hello"
    }
}

日志如下:

4、可观察属性 Observable 

observable 可以用于实现观察者模式。

Delegates.observable() 函数接受两个参数: 第一个是初始化值, 第二个是属性值变化事件的响应器(handler)。

在属性赋值后会执行事件的响应器(handler),它有三个参数:被赋值的属性、旧值和新值:

import kotlin.properties.Delegates

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val user = User()
        user.name="第一次赋值"
        user.name="第二次赋值"
    }
}

class User{
    var name:String by Delegates.observable("初始值"){
        property, oldValue, newValue ->
        println("旧值:$oldValue -> 新值:$newValue")
    }
}

日志如下:

 5、map/mutableMap

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val map = mapOf<String, String>("name" to "xiaoming", "name2" to "xiaoming2")
        val map2 = mutableMapOf<String, String>("name3" to "xiaoming3", "name4" to "xiaoming4")
        map2.put("name5", "xiaoming5")

        println(map["name"])
        println(map["name2"])
        println(map2["name3"])
        println(map2["name4"])
        println(map2["name5"])
    }
}

日志如下:

 十三、kotlin 回调 block:

1、block: () -> T    block: (T) -> T   block: T.() -> T

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        "==1==".method1 {
            println("==1==")
        }

        "==2==".method2 {
            println("==2==this:$it")
            it
        }

        "==3==".method3 {
            println("==3==this:$this")
            this
        }
    }

    /**
     *  1. block: () -> T
     *    01.传入的函数无参
     *    02.该函数最后一行需要是调用者对象类型,而且无return
     */
    fun <T> T.method1(block: () -> T): T {
        return block()
    }

    /**
     *  2. block: (T) -> T
     *    01.传入的函数带有自身作为参数
     *    02.该函数最后一行需要是调用者对象类型,而且无return
     *    03.把调用者作为it,传入定义的lambda表达式函数域中
     */
    fun <T> T.method2(block: (T) -> T): T {
        return block(this)
    }

    /**
     * 3. block: T.() -> T
     *   01.传入的函数无参
     *   02.该函数最后一行需要是调用者对象类型,而且无return
     *   03.把调用者作为this,传入定义的lambda表达式函数域中
     */
    fun <T> T.method3(block: T.() -> T): T {
        println("=======3==this:$this")
        return block()
    }
}

日志如下:

2、Block语法格式

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        test {
            println("test:无返回值")
        }
        test2 {
            "test2:这是返回值"//不需要写 return
        }

        test3 {
            println("test3:${it}")
        }
        test4 {
            println("test4:${it}")
            "test4:这是返回值"//不需要写 return
        }

        test5 { x, y ->
            x + y//不需要写 return
        }
    }

    //1、无参 无返回值
    fun test(block: () -> Unit) {
        return block()
    }

    //2、无参 有返回值
    fun test2(block: () -> String) {
        println(block())
    }

    //3、有参 无返回值
    fun test3(block: (s: String) -> Unit) {
        block("这是入参")
    }

    //4、有参 有返回值
    fun test4(block: (s: String) -> String) {
        println(block("这是入参"))
    }

    //5、有多个参数,有返回值
    fun test5(block: (x: Int, y: Int) -> Int) {
        println(block(5, 10))
    }
}

日志如下:

十四、Kotlin中let、run、with、apply及also的差别

作用域函数是Kotlin比较重要的一个特性,共分为以下5种:let、run、with、apply 以及 also。它们的唯一目的是在对象的上下文中执行代码块。当对一个对象调用这样的函数并提供一个 lambda 表达式时,它会形成一个临时作用域。在此作用域中,可以访问该对象而无需其名称。这些函数称为作用域函数。

1、let  public inline fun <T, R> T.let(block: (T) -> R): R 

1)作用域  let 有入参,有返回值it  

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        //let 作用于 ,直接用 it ,有返回值
        val user = User().let {
            it.name = "xiao ming"
            it.age = 20
            "这是返回值R"
        }
        val user2 = User().let {
            it.name="zhang san"
        }

        println(user)
        println(user2)
    }
}

class User() {
    var name: String? = null
    var age: Int? = null

    fun say() {
        println("name :${name}, age :${age}")
    }
}

日志如下:

 2) 空安全检测  xxx?.let{ }

object Main {

    val name: String? = null

    @JvmStatic
    fun main(args: Array<String>) {
        val result = name?.let {
            it.length
        } ?: "name 空处理"
        println(result)
    }
}

日志如下:

 2、run “this”作为上下文对象,且它的调用方式与let一致。

表达式同时包含对象初始化返回值的计算时,run更适合

public inline fun <T, R> T.run(block: T.() -> R): R 

public inline fun <R> run(block: () -> R): R

无返回值

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val user = User()
        user.run{
            this.name = "xiaoming"
            this.age = 20
            say()
        }
        println(user)
    }
}

class User() {
    var name: String? = null
    var age: Int? = null

    fun say() {
        println("name :${name}, age :${age}")
    }
}

日志如下:

3、with  public inline fun <T, R> with(receiver: T, block: T.() -> R): R 

with属于非扩展函数,直接输入一个对象receiver,当输入receiver后,便可以更改receiver的属性,同时,它也与run做着同样的事情。

with使用的是非null的对象,当函数块中不需要返回值时,可以使用with。

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val user = User()
        with(user) {
            this.name = "xiaoming"
            this.age = 20
            say()
        }
        println(user)
    }
}

class User() {
    var name: String? = null
    var age: Int? = null

    fun say() {
        println("name :${name}, age :${age}")
    }
}

日志如下:

 4、apply public inline fun <T> T.apply(block: T.() -> Unit): T

apply函数主要用于初始化或更改对象,因为它用于在不使用对象的函数的情况下返回自身。

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val user = User()
        user?.apply{
            this.name = "xiaoming"
            this.age = 20
        }
        println(user)
    }
}

class User() {
    var name: String? = null
    var age: Int? = null

    fun say() {
        println("name :${name}, age :${age}")
    }
}

日志如下:

5、also public inline fun <T> T.also(block: (T) -> Unit): T 

also是 T 的扩展函数,返回值与apply一致,直接返回T。also函数的用法类似于let函数,将对象的上下文引用为“it”而不是“this”以及提供空安全检查方面。

object Main {

    @JvmStatic
    fun main(args: Array<String>) {
        val user = User()
        user?.also {
            it.name = "xiaoming"
            it.age = 20
        }
        println(user)
    }
}

class User() {
    var name: String? = null
    var age: Int? = null

    fun say() {
        println("name :${name}, age :${age}")
    }
}

日志如下:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/160100.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

AMD出招,英特尔最不想看到的对手来了

前段时间的CES上&#xff0c;AMD正式发布Ryzen 7000的3D缓存版&#xff0c;对于游戏玩家来说&#xff0c;Ryzen 7000 3D缓存版算是今年最期待的CPU。上一代的Ryzen7 5800X3D凭借超强的游戏性能和性价比&#xff0c;在德国最大的PC硬件零售商的统计中&#xff0c;甚至成为2022年…

高并发系统设计 -- 大文件业务

上传 分片断点秒传&#xff08;判断文件哈希值&#xff09; 前端不断的发送请求&#xff0c;如果用户暂停上传的话&#xff0c;那么就是前端停止发送请求就可以了。我分片了&#xff0c;而且记录了分片的相关信息&#xff0c;所以实现了断点功能。 前端把文件进行分片&#…

ftp vsftp 登录

打开windows资管管理器&#xff08;文件夹&#xff09;输入目标路径&#xff0c;如&#xff1a;ftp://192.168.1.1输入账号密码。 删除用户已保存的密码&#xff08;仅密码&#xff0c;名称记录还在&#xff09; 两种方法都可以试试&#xff0c;适用不同情况 情况-方法一&am…

Set、Map、类数组,傻傻区分不清楚?

前言 大家都知道&#xff0c;数组和对象是两种不同的数据结构&#xff0c;虽说在js数据类型中都属于Object&#xff0c;但是还是有一定的区别&#xff0c;通过字面量以及isArray、instanceof等方法&#xff0c;我们很好区分这两者。由于使用场景的原因js中衍生了很多类似的数据…

基于java(springboot+mybatis)网上音乐商城设计和实现以及论文报告

基于java(springbootmybatis)网上音乐商城设计和实现以及论文报告 博主介绍&#xff1a;5年java开发经验&#xff0c;专注Java开发、定制、远程、文档编写指导等,csdn特邀作者、专注于Java技术领域 作者主页 超级帅帅吴 Java毕设项目精品实战案例《500套》 欢迎点赞 收藏 ⭐留言…

Spring Boot 热部署

Spring Boot 热部署一、添加热部署框架支持二、Settings 开启项目自动编译三、开启运行中热部署四、使用 Debug 启动 (非 Run)一、添加热部署框架支持 或者右击鼠标添加依赖&#xff1a; 或者使用插件&#xff1a; 二、Settings 开启项目自动编译 三、开启运行中热部署 老版…

【数据库概论】第一章 绪论

第一章 绪论 1.1 数据库系统概述 数据库的四个基本概念 1.数据 数据是数据库中存储的基本对象&#xff0c;一般数据是描述事物的符号记录&#xff0c;这种符号记录可以输数字&#xff0c;也可以是文字、徒刑、音频等。 2.数据库 数据库是长期存储在计算机内有组织的&…

Leetcode动态规划题解

第一题 509. 斐波那契数 题目描述&#xff1a;斐波那契数&#xff08;通常用 F(n) 表示&#xff09;形成的序列称为斐波那契数列 。该数列由 0 和 1 开始&#xff0c;后面的每一项数字都是前面两项数字的和。也就是&#xff1a; F(0) 0&#xff0c;F(1) 1 F(n) F(n - 1) …

【计算机网络】计算机网络基础

计算机是人类社会不可或缺的工具&#xff0c;而单独的一台计算机的功能也是有限的&#xff0c;计算机需要和其它的设备相互连接通信形成的计算机网络才能对人类发展带来巨大的影响。 目录 计算机网络 通信协议 网络结构 网络边缘 接入网 网络核心 时延和吞吐量 时延 吞…

.Net Core6.0项目发布在IIS上访问404的问题

ASP.Net Core6.0项目发布在IIS上访问404的问题 进入线程池画面&#xff0c;将当前程序的线程池设为“无托管代码” 修改配置文件 Web.config&#xff0c;以下缺一不可 需要引用架包&#xff1a;Swashbuckle.AspNetCore.SwaggerUI.NetCore 6.0 自带集成了Swagger , 在发布项目时…

C++模板(函数模板、类模板)

目录 一、泛型编程 二、函数模板 函数模板概念 函数模板格式 函数模板的原理 函数模板的实例化 模板参数的匹配原则 三、类模板 类模板的定义格式 类模板的实例化 四、扩展 函数模板一定是推演&#xff1f;类模板一定是指定&#xff1f; 模板的分离编译 一…

MySQL高级【行级锁】

1&#xff1a;行级锁1.1&#xff1a;介绍行级锁&#xff0c;每次操作锁住对应的行数据。锁定粒度最小&#xff0c;发生锁冲突的概率最低&#xff0c;并发度最高。应用在 InnoDB存储引擎中。 InnoDB的数据是基于索引组织的&#xff0c;行锁是通过对索引上的索引项加锁来实现的&a…

WPF中Binding数据校验、并捕获异常信息的三种方式

Binding数据校验、并捕获异常信息的三种方式 WPF在使用Binding时&#xff0c;经常需要进行数据校验&#xff0c;如果校验失败需要捕获失败的原因&#xff0c;并加以展示&#xff0c;本文主要介绍数据校验异常并捕获的三种方式。 依赖属性异常捕获 先定义一个依赖属性 publi…

【Nacos】Nacos配置中心的使用与SpringCloud整合

在微服务架构中&#xff0c;当系统从一个单体应用&#xff0c;被拆分成分布式系统上一个个服务节点后&#xff0c;配置文件也必须跟着迁移&#xff08;分割&#xff09;&#xff0c;这样配置就分散了&#xff0c;不仅如此&#xff0c;分散中还包含着冗余。配置中心将配置从各应…

哪儿有微服务开源项目?

随着数字化时代的到来&#xff0c;微服务开源项目的应用价值逐渐凸显。作为提升企业办公协作效率的低代码开发平台项目&#xff0c;其表现出来的灵活性、易操作、简便的特性&#xff0c;成为现代化办公管理中的重要合作伙伴。我们今天一起来了解什么是微服务开源项目。 一、微服…

基于JavaWeb实现蜀南调味品商城物流配货系统

一、项目介绍 本文系统利用JavaWeb技术&#xff0c;设计和实现了连接公司、客户公司、物流运输为桥梁的销售配送管理系统&#xff0c;并以网络技术和信息技术在销售配送中的应用为重点&#xff0c;实现员工登录模块、员工信息管理模块、库存管理模块、订单处理模块、包装管理模…

高通Wi-Fi 7网络芯片方案IPQ9574,IPQ9554,IPQ9514,IPQ9570,IPQ9550,IPQ9510

networking pro 1620&#xff1a;芯片型号IPQ9574&#xff0c;支持4频段16路数据流&#xff0c;峰值速率33Gbps&#xff0c;支持4个2.5G口&#xff0c;1个5G口&#xff0c;1个万兆口&#xff1b;networking pro 1220&#xff1a;芯片型号IPQ9574&#xff0c;支持3频段12路数据流…

三个案例详解不同网段之间如何互通

当然还可以通过三层交换机划分VLAN配置更好。这里主要讲通过普通路由器之间互通一台路由器连接另外一台路由器&#xff0c;这两台路由器分别连接不同的网段&#xff0c;那么如果要这两个网段互通&#xff0c;则必须配置路由&#xff0c;这个就是静态路由。案例一、不同网段之间…

【练习】Day06

努力经营当下&#xff0c;直至未来明朗&#xff01; 文章目录一、选择二、编程最小时间差答案1. 选择2. 编程普通小孩也要热爱生活&#xff01; 一、选择 散列技术中的冲突是指&#xff08; &#xff09; A. 两个元素具有相同的序号 B. 两个元素的键值不同&#xff0c;而其他…

Linux权限理解

✅<1>主页&#xff1a;我的代码爱吃辣 &#x1f4c3;<2>知识讲解&#xff1a;C ☂️<3>开发环境&#xff1a;Visual Studio 2022 &#x1f4ac;<4>前言&#xff1a;linux当中对于权限的理解。 &#x1f490;一.生活中的权限 &#x1f338;二.Linux权限…