首页 > 其他分享 >Android Kotlin实战之高阶使用泛型扩展协程懒加载详解

Android Kotlin实战之高阶使用泛型扩展协程懒加载详解

时间:2023-04-12 10:31:45浏览次数:54  
标签:协程 String val Kotlin fun kotlin 泛型 class name


前言:

通过前面几篇文章,我们已基本掌握kotlin的基本写法与使用,但是在开发过程中,以及一些开源的API还是会出现大家模式的高阶玩法以及问题,如何避免,接下来讲解针对原来的文章进行一些扩展,解决大家在工作中遇到的问题,如何去解决

如果还有人不了解kotlin,可以查看我的基础篇kotlin。

Android kotlin在实战过程问题总结与开发技巧详解_kotlin 同步锁_蜗牛、Z的博客-

Kotlin语法详解与实践教程,区分JAVA以及如何闭坑_kotlin mapof可以嵌套 to_蜗牛、Z的博客-

1. lazy

lazy是kotlin中的懒加载,这种写法在很多场景中都有,懒加载并不是立刻在内存中申请,而是通过lazy(),调用才会

lazy的只能是常量,用val修饰,变量是全局也可以是局部变量

Android Kotlin实战之高阶使用泛型扩展协程懒加载详解_密封类

通过这个lazy{}可以看出,无论你在最上面调用什么,默认值都是最后一个。通过IDE,可以提示看到ccc ^lazy,这就是懒加载的值。这个值也可以通过方法获取

fun a() {
    val aa: String by lazy {
        MyLog.log("val aa:String by lazy start")
        "aaaa"
        "bbbb"
        b()

    }

    MyLog.log(aa)
    MyLog.log(aa)
}


fun b(): String {
    return "aaaaa"
}

一旦懒加载对象被调用过,后期在调用将只能取到值,不会再处理其他的lazy{} 方法体。

Android Kotlin实战之高阶使用泛型扩展协程懒加载详解_android studio_02

注意:lazy和lateinit无法一起使用。lazy修饰val变量,lateinit修饰的是var变量。

2.lateinit

也是懒加载机制,对象是var的格式。但是,这个变量需要是全局变量,不能是局部变量。

lateinit var string: String
lateinit var people: People
fun c() {

    people = People()
    people.name = "zhangshan"

    string = "aaaa"
    MyLog.log(people.name!!)

    val child: People by lazy{
        People()
    }
    

}

3.如何避免参数加"!!"可为空符号

kotlin的变量如果不是懒加载修饰,那么你在申请的时候,就要给出变量值,即默认值

常见写法:

var name:String?=null,表示当前对象是null


var title:String="title",表示已默认值、


但是如果你申请为null,那么你在赋值的时候,这个参数就要在多个地方加上"!!",否则会报错。这种写法很烦人。

如何避免?

使用懒加载机制,通过上面两个懒加载,可以避免这种问题的参数。否则你在任何地方进程参数传递都会被判为null。

4.类的扩展函数

kotlin支持在现有的类中,动态扩展函数,格式就是类对象+"."+函数名,可以支持参数和返回值,扩展的函数和正常函数一样使用

以下是对String扩展一个两个方法,分别是log()和size()

fun String.log(log:String) {

    MyLog.log(log)
}

fun String.size(): Int {

    return toString().length
}

5.扩展属性、扩展变量

kotlin支持动态新增变量,格式就类名+"."+变量名+":"+类型

这种写法可以无限的扩展你需要的变量,扩展变量一样,只能在最外围,不能在方法体中定义,

扩展属性需要额外重写get()方法,且不能申请直接赋值

val String.defaulename: String
    get() {
        return "defaulename"
    }


fun e() {
    val name=""
    MyLog.log(name.defaulename)
}

这里需要重写get(),给出默认值。

6.apply与let与also

apply与alse是链式设置,返回的是当前对象,let返回值是unit,

  • apply、also,闭包的返回值都是this,前者apply接受的闭包类型调用者的扩展函数,后者接受的闭包类型为 入参为调用者类型的函数;
  • also、apply,非常适合对同一个对象连续操作的链式调用;
  • run、let,闭包的返回值为最后一行非赋值代码,前者run接受的闭包类型调用者的扩展函数,后者接受的闭包类型为 入参为调用者类型的函数;
  • run、let,非常适合上一个操作返回值作用于下一个操作的调用;

我们经常会使用如下写法:

if(people!=null){

peope.name="zhangshan"

}

apply写法:

people?.apply{it->

it.name="zhangsnan"

}

7.类的初始化模块init

kotlin中没有static{}模块,是通过init{}替代了static。如果需要提前初始化的可以放在init中,常见的so库加载可以放到init中

class Example {
    init {

        //init module
    }
}

8 .接口多继承,方法名冲突

 如果接口中有两个相同的方法名,在java中是需要修改一个的。在kotlin中就可以避免这种。只要在继承方法中,指向各个类的即可。

Android Kotlin实战之高阶使用泛型扩展协程懒加载详解_密封类_03

class TestInterface : IntFacA, IntFacB {

    override fun log() {
        super<IntFacA>.log()
        super<IntFacB>.log()
    }

    override fun info() {

    }
}

9.internal 介绍

internal 属于修饰,和provite、public、protect一样

  • private 意味着只在这个类内部(包含其所有成员)可见;
  • protected—— 和 private一样 + 在子类中可见。
  • internal —— 能见到类声明的 本模块内 的任何客户端都可见其 internal 成员;
  • public —— 能见到类声明的任何客户端都可见其 public 成员。

10.数据类(data)

将类对象修饰成data,需要的变量直接在构造器中申明,不需要方法体

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

11.密封类(sealed)

密封类用来表示受限的类继承结构:当一个值为有限几种的类型、而不能有任何其他类型时。在某种意义上,他们是枚举类的扩展:枚举类型的值集合也是受限的,但每个枚举常量只存在一个实例,而密封类的一个子类可以有可包含状态的多个实例。

要声明一个密封类,需要在类名前面添加 sealed 修饰符。虽然密封类也可以有子类,但是所有子类都必须在与密封类自身相同的文件中声明

sealed class Expr
data class Const(val number: Double) : Expr()
data class Sum(val e1: Expr, val e2: Expr) : Expr()
object NotANumber : Expr()

一个密封类是自身抽象的,它不能直接实例化并可以有抽象(abstract)成员。

密封类不允许有非-private 构造函数(其构造函数默认为 private)。

fun eval(expr: Expr): Double = when(expr) {
    is Const -> expr.number
    is Sum -> eval(expr.e1) + eval(expr.e2)
    NotANumber -> Double.NaN
    // 不再需要 `else` 子句,因为我们已经覆盖了所有的情况
}

类似接口继承,通过泛型来判断

12.泛型,通配符类型参数:in 和 out

在开发过程中,泛型的利用,可以很好的解决代码臃肿问题,也能很好的进行解耦。在kotlin中,泛型的使用有普通的,也有通配符限定

1.正常泛型

interface Factory<T> {


    fun getEntity(): T
}

class Room : Factory<String> {

    override fun getEntity(): String {
        return "String info"
    }
}

这种是我们最常见的泛型,通过泛型,可以很好的获取不同的对象类型

2通配符 out和in

java中也有通配符,? extend E,? extends E 表示此方法接受 E 或者 E 的 一些子类型对象的集合,而不仅仅是 E 自身。简而言之,带 extends 限定(上界)的通配符类型使得类型是协变的。

还有一种,? super String,表示只接收String类型。

在 kotlin 语言中,out 表示协变,in 表示逆变;

kotlin 中的 “out T” 等同于 Java 的 “?extends T”

kotlin 中的 “in T” 等同于 Java 的 “?super T”

也就是上线和下线。

13.嵌套类

内部类可以直接使用,就是通过最外层类引用即可,也可以单独使用

class Student {

    lateinit var name: String

    class Classmate {

        lateinit var name: String

    }
}


fun main() {

    var mate=Student.Classmate()
    mate.name="zhangsan"
    MyLog.log(mate.name)
}

14.内部类:inner 

标记为 inner 的嵌套类能够访问其外部类的成员。内部类会带有一个对外部类的对象的引用。

class Student {

    lateinit var name: String

   inner class Classmate {

        lateinit var name: String

    }
}


fun main() {

    var mate=Student().Classmate()
    mate.name="zhangsan"
    MyLog.log(mate.name)
}

15.匿名内部类:

使用对象表达式创建匿名内部类实例:

window.addMouseListener(object : MouseAdapter() {

    override fun mouseClicked(e: MouseEvent) { …… }

    override fun mouseEntered(e: MouseEvent) { …… }
})

接口的监听:

1.lambda表达式

Android Kotlin实战之高阶使用泛型扩展协程懒加载详解_kotlin_04

2.object匿名内部类

view.setOnClickListener(object :View.OnClickListener{
        override fun onClick(v: View?) {

        }
    })

16.枚举类

枚举类的最基本的用法是实现类型安全的枚举,kotlin枚举有一下几种

1.支持匿名类

2.支持参数扩展

enum class MyEmu {
    ONE{
         var size="123"
       },TWO,THREE
}

enum class MyEmu1(tag:Int) {
    ONE(1),TWO(2),THREE(3)
}

如果你想弄那么复杂,就当普通枚举使用

3、枚举类型新增接口

enum class MyEmu :MyInfo{
    ONE{
        override fun log() {
            TODO("Not yet implemented")
        }
    },TWO{
        override fun log() {
            TODO("Not yet implemented")
        }
    }
}
interface MyInfo {

    fun log();
}

如果枚举类型实现了接口,那么枚举中的任何元素的都变成了匿名内部类,都要实现接口的方法

17.open超类

如果一个类无法被继承,就可以将该类通过open修饰,如果类中的变量无法使用,也可以通过open修饰。

open class MyNAME {
    open var name = ""
}

18.静态类:object 

静态类,是通过object修饰,该类里面的所有变量和方法都是静态。且不在需要class修饰

object staticName {
    var name = ""

    fun log() {
        
    }
}

19.伴生对象

伴随对象是在普通类中进行扩展一个对象出来的,该伴生对象的成员可通过只使用类名作为限定符来调用

class BanshengObject {

    companion object {
        var name = ""

        fun log() {

        }
    }
}


fun main() {

    BanshengObject.log()
    BanshengObject.name
}

请注意,即使伴生对象的成员看起来像其他语言的静态成员,在运行时他们仍然是真实对象的实例成员

如果使用 @JvmStatic 注解,你可以将伴生对象的成员生成为真正的静态方法和字段

20.类型别名:typealias

        类型别名为现有类型提供替代名称。 如果类型名称太长,你可以另外引入较短的名称,并使用新的名称替代原类型名。

class Alias {

    inner class MyStudent(name:String)
}

typealias Alias_MyStudent = Alias.MyStudent

typealias SetString=Set<String>
typealias MyHandler = (Int, String, Any)->Void

fun main() {

    var m:MyHandler

   

}

这个别名使用在kotlin有很多,kotlin对java的api进行封装,就是通过别名完成。也可以支持方法的别名

21.内联类

内联类必须含有唯一的一个属性在主构造函数中初始化。在运行时,将使用这个唯一属性来表示内联类的实例

inline class InlinerBean(val title:String) {

    val length: Int
        get() = title.length
}

fun main() {
    val  bean=InlinerBean("this is inline class")
    
}

// 不存在 'InlinerBean' 类的真实实例对象
// 在运行时,'InlinerBean' 仅仅包含 'String' 

 val  bean=InlinerBean("this is inline class")

同时,内联是为了解决额外的堆内存分配问题。

内联类支持普通类中的一些功能。特别是,内联类可以声明属性与函数

内联函数,有且仅有一个构造参数

内联函数也可以当普通函数使用,可以实现接口

22.委托:by

委托模式已经证明是实现继承的一个很好的替代方式, 而 Kotlin 可以零样板代码地原生支持它

interface Base {
    val message: String
    fun print()
}

class BaseImpl(val x: Int) : Base {
    override val message = "BaseImpl: x = $x"
    override fun print() { println(message) }
}

class Derived(b: Base) : Base by b {
    // 在 b 的 `print` 实现中不会访问到这个属性
    override val message = "Message of Derived"
}

fun main() {
    val b = BaseImpl(10)
    val derived = Derived(b)
    derived.print()
    println(derived.message)
}

注意,以这种方式重写的成员不会在委托对象的成员中调用 ,委托对象的成员只能访问其自身对接口成员实现。

23.委托属性

有一些常见的属性类型,虽然我们可以在每次需要的时候手动实现它们, 但是如果能够为大家把他们只实现一次并放入一个库会更好。例如包括:

  • 延迟属性(lazy properties): 其值只在首次访问时计算;
  • 可观察属性(observable properties): 监听器会收到有关此属性变更的通知;
  • 把多个属性储存在一个映射(map)中,而不是每个存在单独的字段中。

为了涵盖这些(以及其他)情况,Kotlin 支持 委托属性:

class Example {
    var p: String by Delegate()
}

语法是: val/var <属性名>: <类型> by <表达式>。在 by 后面的表达式是该 委托, 因为属性对应的 get()(与 set())会被委托给它的 getValue() 与 setValue() 方法。 属性的委托不必实现任何的接口,但是需要提供一个 getValue() 函数(与 setValue()——对于 var 属性)

class Delegate {
    operator fun getValue(thisRef: Any?, property: KProperty<*>): String {
        return "$thisRef, thank you for delegating '${property.name}' to me!"
    }
 
    operator fun setValue(thisRef: Any?, property: KProperty<*>, value: String) {
        println("$value has been assigned to '${property.name}' in $thisRef.")
    }
}

分析:

by是ReadOnlyProperty的连接,

public fun interface ReadOnlyProperty<in T, out V> {
    /**
     * Returns the value of the property for the given object.
     * @param thisRef the object for which the value is requested.
     * @param property the metadata for the property.
     * @return the property value.
     */
    public operator fun getValue(thisRef: T, property: KProperty<*>): V
}

所有我们在属性委托时,需要实现 public operator fun getValue(thisRef: T, property: KProperty<*>): V

所以定义属性委托必须要实现接口ReadOnlyProperty

实战:

在jetpack的datastore中,也提供了属性委托。

val Context.dataStore: DataStore<Preferences> by preferencesDataStore(
        "my_datastore",
        produceMigrations = { it ->
            listOf(SharedPreferencesMigration(it, "sp_test"))
        })

24.Lambda 表达式

lambda 表达式与匿名函数是“函数字面值”,即未声明的函数, 但立即做为表达式传递

max(strings, { a, b -> a.length < b.length })

函数 max 是一个高阶函数,它接受一个函数作为第二个参数。 其第二个参数是一个表达式,它本身是一个函数,如下:

fun compare(a: String, b: String): Boolean = a.length < b.length

Lambda 表达式语法

Lambda 表达式的完整语法形式如下:

val sum: (Int, Int) -> Int = { x: Int, y: Int -> x + y }

lambda 表达式总是括在花括号中, 完整语法形式的参数声明放在花括号内,并有可选的类型标注, 函数体跟在一个 -> 符号之后

val sum = { x: Int, y: Int -> x + y }
val sum: (Int, Int) -> Int = { x: Int, y: Int -> x + y }
    sum(2,3)

    val add = { x: Int, y: Int -> x + y }
    add(2,3)

从 lambda 表达式中返回一个值

ints.filter {
    val shouldFilter = it > 0 
    shouldFilter
}

ints.filter {
    val shouldFilter = it > 0 
    return@filter shouldFilter
}

25.构造集合

在开发过程中,发现使用集合都需要指定泛型,还有一种可以指定一个默认值,后面所有的集合都是这种

val numbersSet = setOf("one", "two", "three", "four")
val emptySet = mutableSetOf<String>()

26.空集合

还有用于创建没有任何元素的集合的函数:emptyList()、emptySet() 与 emptyMap()。 创建空集合时,应指定集合将包含的元素类型。

27.具体类型构造函数

要创建具体类型的集合,例如 ArrayList 或 LinkedList,可以使用这些类型的构造函数。 类似的构造函数对于 Set 与 Map 的各实现中均有提供

val linkedList = LinkedList<String>(listOf("one", "two", "three"))
val presizedSet = HashSet<Int>(32)

28.集合加减法

我们正常使用逻辑都是对于变量运算,但是在kotlin中的集合也可以使用

val plusList = numbers + "five"
    val minusList = numbers - listOf("three", "four")
    val minusList2 = numbers - "three"
    println(plusList)
    println(minusList)
    println(minusList2)

Android Kotlin实战之高阶使用泛型扩展协程懒加载详解_android studio_05

29.集合数据分组

Kotlin 标准库提供用于对集合元素进行分组的扩展函数。 基本函数 groupBy() 使用一个 lambda 函数并返回一个 Map。 在此 Map 中,每个键都是 lambda 结果,而对应的值是返回此结果的元素 List。 例如,可以使用此函数将 String 列表按首字母分组。 

val numbers = listOf("one", "two", "three", "four", "five")

    println(numbers.groupBy { it.first().toUpperCase() })

Android Kotlin实战之高阶使用泛型扩展协程懒加载详解_密封类_06

30.协程

Kotlin 是一门仅在标准库中提供最基本底层 API 以便各种其他库能够利用协程的语言。与许多其他具有类似功能的语言不同,async 与 await 在 Kotlin 中并不是关键字,甚至都不是标准库的一部分。此外,Kotlin 的 挂起函数 概念为异步操作提供了比 future 与 promise 更安全、更不易出错的抽象。

kotlinx.coroutines 是由 JetBrains 开发的功能丰富的协程库。它包含本指南中涵盖的很多启用高级协程的原语,包括 launch、 async 等等。

 

在kotlin中,suspend修饰的方法叫着协程。目前针对协程有一下几种

1.同步:runBlocking:

2.异步:kotlinx-coroutines-core库

 代码库:

implementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.1"
class MySuspendMain {
    suspend fun info() {

    }
}


fun main() {
    GlobalScope.launch { // 在后台启动一个新的协程并继续
        delay(1000L) // 非阻塞的等待 1 秒钟(默认时间单位是毫秒)
        println("World!") // 在延迟后打印输出
    }
    println("Hello,") // 协程已在等待时主线程还在继续
    Thread.sleep(2000L) // 阻塞主线程 2 秒钟来保证 JVM 存活

    runBlocking {
        val mains=MySuspendMain()
        mains.info()

    }
    

}

由于kotlinx-coroutines-core库很大,我会出一篇详细的文章讲解。

Android kotlin实战之协程suspend详解与使用_蜗牛、Z的博客-


标签:协程,String,val,Kotlin,fun,kotlin,泛型,class,name
From: https://blog.51cto.com/u_16065093/6185195

相关文章

  • 一文吃透泛型
    本文已经收录到Github仓库,该仓库包含计算机基础、Java基础、多线程、JVM、数据库、Redis、Spring、Mybatis、SpringMVC、SpringBoot、分布式、微服务、设计模式、架构、校招社招分享等核心知识点,欢迎star~Github地址如果访问不了Github,可以访问gitee地址。gitee地址Java泛型(g......
  • Android Kotlin mapTo
     在Kotlin中,mapTo是一种用于将集合中的元素转换成另一个集合的函数。它可以将一个集合的元素映射到另一个集合,并将结果添加到目标集合中。mapTo的语法如下:fun<T,R,C:MutableCollection<inR>>Iterable<T>.mapTo(destination:C,transform:(T)->R):C其中:T是源集......
  • Kotlin Compose 删除条目并刷新数据源 Flow 与Pager3 与 Compose 最佳做法
    删除方式有很多种。。成功的却不多。。分享一下首先在viewModel中创建两个变量privateval_deletedDialogHashs=mutableStateOf(setOf<String>())valdeletedDialogHashs:State<Set<String>>=_deletedDialogHashs在viewModel中声明的删除方法也很简单funremoveFeedDial......
  • 【学习笔记】go协程和通道
    虽然,线程池为逻辑编写者提供了线程分配的抽象机制。但是,如果面对随时随地可能发生的并发和线程处理需求,线程池就不是非常直观和方便了。能否有一种机制:使用者分配足够多的任务,系统能自动帮助使用者把任务分配到CPU上,让这些任务尽量并发运作。这种机制在Go语言中被称为goroutine。go......
  • 生成器、协程
    生成器、协程目录生成器、协程1协程和生成器2生成器Generator2.1列表生成式2.2生成器2.3斐波拉契数列(Fibonacci)2.3.1斐波拉契数列函数写法2.3.2yield方式生成斐波拉契数列函数2.4yield生成器返回值特点2.5yieldfrom3协程3.1协程介绍3.1.1存在yield函数运行过程3.1......
  • kotlin minOf 发出NoSuchElementException异常,该如何解决
    kotlinminOf发出NoSuchElementException异常,该如何解决minOf函数在处理空集合时会抛出NoSuchElementException异常。解决该问题的方法取决于你的具体需求和代码实现方式。如果你期望集合可能为空,你可以使用minOrNull函数代替minOf函数。minOrNull函数返回集合中的最小......
  • 19.协程 - 2
    协程-asyncio-2异步编程asyncio.Future对象Task继承Future,Task对象内部中的await结果的处理基于Future对象来的在Future对象中会保存当前执行的这个协程任务的状态,如果当前任务状态为finished,则await不再等待。示例1:importasyncioasyncdefmain():#......
  • java -- 异常处理、Collection、Iterator迭代器、泛型
    异常处理Java异常处理的五个关键字:try、catch、finally、throw、throws抛出异常throw在编写程序时,我们必须要考虑程序出现问题的情况当调用方法使用接受到的参数时,首先需要先对参数数据进行合法的判断,数据若不合法,就应该告诉调用者,传递合法的数据进来。这时需要使用抛出异常的......
  • 泛型
    泛型泛型在创建对象时指定泛型类型在创建泛型类时若没有指定泛型类型,将按照Object类型处理不支持基本数据类型泛型类派生子类泛型接口泛型接口得使用实现类不是泛型类,接口要明确数据类型实现类也是泛型类,实现接类与接口泛型类型要一致类型通配符当参数传递......
  • C#泛型
    泛型(Generic)允许您延迟编写类或方法中的编程元素的数据类型的规范,直到实际在程序中使用它的时候。换句话说,泛型允许您编写一个可以与任何数据类型一起工作的类或方法。您可以通过数据类型的替代参数编写类或方法的规范。当编译器遇到类的构造函数或方法的函数调用时,它会生成代码......