Kotlin Multiplatform--04:经验总结(持续更新)

Kotlin Multiplatform--04:经验总结(持续更新)

引言

本章用来记载笔者开发过程中的一些经验总结


一、Ktor设置Header

官方文档中,想要设置Header的示例代码如下:

kotlin 复制代码
client.get("https://ktor.io") {
    headers {
        append(HttpHeaders.Accept, "text/html")
        append(HttpHeaders.Authorization, "abc123")
        append(HttpHeaders.UserAgent, "ktor client")
    }
}

但经过笔者多次测试和踩坑,在最新版本的Ktor中,这样设置Header根本没用,需要用以下代码设置:

kotlin 复制代码
client.get("https://ktor.io") {
    headers["User-Agent"] = "ktor client"
}

二、IO与ViewModel

ViewModel是Android开发中一个很重要的概念,因为它的存活时间相较于StateHodler更长,所以最适合用来处理业务逻辑。普通的ViewModel不能有任何参数,如下所示:

kotlin 复制代码
class MyViewModel: ViewModel() { ... }

由于ViewModel存活时间长,因此它不能直接引用Context对象,更不能被其他对象引用。定义了自己的MyViewModel类后,通过以下函数引用MyViewModel对象,哪里需要就在哪里调用,如果没有这个函数,则需要添加以下依赖:

kotlin 复制代码
// androidx.lifecycle:lifecycle-viewmodel-compose
val myViewModel: MyViewModel = viewModel()

当viewModel中需要涉及文件读写等IO操作时,可以使用viewModelScope和withContext切换到IO线程,防止阻塞UI。示例代码如下:

kotlin 复制代码
class MyViewModel: ViewModel() {
    fun save(context: Context) {
        viewModelScope.launch {
            withContext(Dispatchers.IO) {
                File(context.getExternalFilesDir(Environment.DIRECTORY_DOCUMENTS), "data.json").outputStream().use {
                    it.write(...)
                }
            }
        }
    }
}

三、测量与布局

IntrinsicSize 用于控制组件的 固有尺寸测量(intrinsic measurement),它允许你在布局过程中获取或使用组件的固有宽度或高度。当需要根据子组件的尺寸来约束父组件大小时,可以使用IntrinsicSize实现。

例如我有一个父组件Box和两个子组件Row,其中包含Text组件的Row大小已知,需要让另一个Row大小一致,此时可以用如下代码实现:

kotlin 复制代码
 Box(Modifier.height(IntrinsicSize.Min)) {
     Row {
         Text(value, fontSize = TextUnit(26f, TextUnitType.Sp))
     }
     Row(Modifier.fillMaxSize()) {
         ...
     }
 }

注意事项:

1、固有尺寸计算可能会有性能开销,应避免在性能敏感的场景过度使用。

2、并非所有组件都支持固有尺寸(如 LazyColumn 就没有固有高度)。

四、回调与协程

回调函数是非常常见的一种编程范式,在Java中尤为突出。但回调函数一多就会形成回调地狱,不管是开发还是维护都十分痛苦。Kotlin为了解决这一痛点,设计了suspendCoroutine。suspendCoroutine 是 Kotlin 协程中的一个核心函数,用于将传统的基于回调的异步操作转换为协程风格的挂起函数。示例代码如下所示:

kotlin 复制代码
// 模拟常见的回调函数
fun someCallbackFun(callback: (String) -> Unit) {
    callback("myCallback")
}

// 正常调用方法
fun callBackMain() {
    someCallbackFun { 
        println(it)
    }
}

// 协程调用,将非线性程序转为线性程序
suspend fun coroutineMain() {
    val retval = suspendCoroutine { coroutine ->
        someCallbackFun {
            coroutine.resume(it)
        }
    }
    print(retval)
}
相关推荐
TA远方34 分钟前
【Android】adb常用的命令用法详解
android·adb·管理·控制·命令
Boilermaker19926 小时前
[Java 并发编程] Synchronized 锁升级
java·开发语言
MM_MS6 小时前
Halcon变量控制类型、数据类型转换、字符串格式化、元组操作
开发语言·人工智能·深度学习·算法·目标检测·计算机视觉·视觉检测
꧁Q༒ོγ꧂7 小时前
LaTeX 语法入门指南
开发语言·latex
njsgcs7 小时前
ue python二次开发启动教程+ 导入fbx到指定文件夹
开发语言·python·unreal engine·ue
alonewolf_997 小时前
JDK17新特性全面解析:从语法革新到模块化革命
java·开发语言·jvm·jdk
古城小栈7 小时前
Rust 迭代器产出的引用层数——分水岭
开发语言·rust
ghie90907 小时前
基于MATLAB的TLBO算法优化实现与改进
开发语言·算法·matlab
恋爱绝缘体17 小时前
2020重学C++重构你的C++知识体系
java·开发语言·c++·算法·junit
wuk9987 小时前
VSC优化算法MATLAB实现
开发语言·算法·matlab