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)
}
相关推荐
Tony Bai1 小时前
“我曾想付钱给 Google 去工作”—— Russ Cox 深度访谈:Go 的诞生、演进与未来
开发语言·后端·golang
sali-tec1 小时前
C# 基于halcon的视觉工作流-章66 四目匹配
开发语言·人工智能·数码相机·算法·计算机视觉·c#
一笑的小酒馆1 小时前
Android CameraX适配Android15
android
hnlgzb1 小时前
安卓app开发,如何快速上手kotlin和compose的开发?
android·开发语言·kotlin
alexhilton2 小时前
Jetpack Compose 2025年12月版本新增功能
android·kotlin·android jetpack
思成不止于此2 小时前
【MySQL 零基础入门】DQL 核心语法(二):表条件查询与分组查询篇
android·数据库·笔记·学习·mysql
无敌最俊朗@2 小时前
STL-deque面试剖析(面试复习4)
开发语言
APIshop2 小时前
用 Python 把“API 接口”当数据源——从找口子到落库的全流程实战
开发语言·python
Java Fans2 小时前
Qt Designer 和 PyQt 开发教程
开发语言·qt·pyqt
RwTo2 小时前
【源码】-Java线程池ThreadPool
java·开发语言