Android面试题之Kotlin协程并发问题和互斥锁

简介: Kotlin的协程提供轻量级并发解决方案,如`kotlinx.coroutines`库。`Mutex`用于同步,确保单个协程访问共享资源。示例展示了`withLock()`、`lock()`、`unlock()`和`tryLock()`的用法,这些方法帮助在协程中实现线程安全,防止数据竞争。

本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点

Kotlin 语言提供了多种机制来处理并发和同步,其中包括高层次和低层次的工具。对于常规的并发任务,可以利用 Kotlin 协程提供的结构化并发方式。而对于需要更低层次的锁定机制,可以使用 Mutex 来实现对共享资源的线程安全访问。

Kotlin 协程与并发(Coroutines and Concurrency)

协程是一种轻量级的线程,可以通过 kotlinx.coroutines 库来实现。协程为结构化并发提供了强大的支持,使得编写异步、并发代码变得更加简单和直观。

协程基础

import kotlinx.coroutines.*

fun main() = runBlocking {
    launch {
        delay(1000L)
        println("World!")
    }
    println("Hello,")
}

在这个例子中,runBlocking 函数用于启动一个新的协程并阻塞当前线程,而 launch 函数则用于启动一个新的协程,并在1秒后输出 "World!"。

并发与同步

当多个协程需要访问共享资源时,需要一些同步机制来防止数据竞争。一个常用的方法是使用 Kotlin 库提供的 Mutex

Mutex

Mutex(互斥锁)是一种用于保证互斥访问共享资源的同步机制。Mutex 确保在同一时刻只有一个协程能够访问被保护的代码块或资源,从而避免竞争条件。

使用 Mutex

import kotlinx.coroutines.*
import kotlinx.coroutines.sync.Mutex
import kotlinx.coroutines.sync.withLock

val mutex = Mutex()
var counter = 0

fun main() = runBlocking {
    val jobs = List(100) {
        launch {
            repeat(1000) {
                // 在这里使用 mutex 来保护对 counter 的访问
                mutex.withLock {
                    counter++
                }
            }
        }
    }
    jobs.forEach { it.join() }
    println("Counter = $counter")
}

在这个例子中,我们创建了100个协程,每个协程重复1000次对共享变量 counter 的访问。使用 mutex.withLock 保证了每次只有一个协程能访问 counter,从而避免并发问题。

withLock() 是一种便捷方法,用于在锁内执行给定的代码块。它会自动处理获取和释放锁,确保即使在代码块中发生异常,也会正确释放锁。

Mutex 的其他方法

lock:挂起直到互斥锁被锁定。

lock() 方法用于尝试获取锁。如果锁已经被其他协程持有,那么调用 lock() 的协程将会被挂起,直到锁变为可用。

用法
import kotlinx.coroutines.*
import kotlinx.coroutines.sync.Mutex

val mutex = Mutex()

fun main() = runBlocking {
    launch {
        mutex.lock() // 获取锁
        try {
            // 保护的代码段
            println("Locked by coroutine 1")
            delay(1000)
        } finally {
            mutex.unlock() // 确保释放锁
        }
    }

    launch {
        mutex.lock() // 等待并获取锁
        try {
            // 保护的代码段
            println("Locked by coroutine 2")
        } finally {
            mutex.unlock() // 确保释放锁
        }
    }
}

unlock:解锁互斥锁。

unlock() 方法用于释放锁,使得被挂起的其他协程可以继续执行。如果 unlock() 被调用时没有持有锁,则会引发异常。

用法

如上面 lock() 示例中的 finally 块所示。

tryLock

tryLock() 尝试获取锁,如果锁当前是可用的,则立即获取锁并返回 true;否则返回 false,且不会挂起当前协程。

用法
import kotlinx.coroutines.*
import kotlinx.coroutines.sync.Mutex

val mutex = Mutex()

fun main() = runBlocking {
    launch {
        if (mutex.tryLock()) { // 尝试获取锁
            try {
                println("Lock acquired by coroutine 1")
                delay(1000)
            } finally {
                mutex.unlock()
            }
        } else {
            println("Coroutine 1: Lock not acquired")
        }
    }

    launch {
        if (mutex.tryLock()) { // 尝试获取锁
            try {
                println("Lock acquired by coroutine 2")
            } finally {
                mutex.unlock()
            }
        } else {
            println("Coroutine 2: Lock not acquired")
        }
    }
}

总结

  • lock():尝试获取锁,如果锁不可用,则挂起当前协程。
  • unlock():释放锁,其他挂起的协程可以继续执行。
  • tryLock():尝试获取锁,如果锁不可用,则立即返回 false,不会挂起当前协程。
  • withLock():便捷方法,自动获取和释放锁,确保在代码块执行后释放锁。

Mutex 的这些方法使得在 Kotlin 协程中进行线程安全的操作变得更加简洁和直观。根据实际需求选择合适的方法,可以有效避免并发问题,提高代码的健壮性和可维护性。


欢迎关注我的公众号AntDream查看更多精彩文章!

目录
相关文章
|
25天前
|
编译器 Android开发 开发者
Android经典面试题之Kotlin中Lambda表达式和匿名函数的区别
Lambda表达式和匿名函数都是Kotlin中强大的特性,帮助开发者编写简洁而高效的代码。理解它们的区别和适用场景,有助于选择最合适的方式来解决问题。希望本文的详细讲解和示例能够帮助你在Kotlin开发中更好地运用这些特性。
29 9
|
3月前
|
缓存 NoSQL 关系型数据库
大厂面试高频:如何解决Redis缓存雪崩、缓存穿透、缓存并发等5大难题
本文详解缓存雪崩、缓存穿透、缓存并发及缓存预热等问题,提供高可用解决方案,帮助你在大厂面试和实际工作中应对这些常见并发场景。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:如何解决Redis缓存雪崩、缓存穿透、缓存并发等5大难题
|
4月前
|
Android开发
Android面试之Activity启动流程简述
Android面试之Activity启动流程简述
111 6
|
4月前
|
Android开发
Android面试高频知识点(1) 图解Android事件分发机制
Android面试高频知识点(1) 图解Android事件分发机制
|
4月前
|
消息中间件 存储 Java
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
|
4月前
|
XML 前端开发 Android开发
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
|
4月前
|
Android开发
Android面试之Activity启动流程简述
Android面试之Activity启动流程简述
33 0
|
4月前
|
JSON 调度 数据库
Android面试之5个Kotlin深度面试题:协程、密封类和高阶函数
本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点。文章详细解析了Kotlin中的协程、扩展函数、高阶函数、密封类及`inline`和`reified`关键字在Android开发中的应用,帮助读者更好地理解和使用这些特性。
68 1
|
5月前
|
Android开发 开发者 Kotlin
告别AsyncTask:一招教你用Kotlin协程重构Android应用,流畅度飙升的秘密武器
【9月更文挑战第13天】随着Android应用复杂度的增加,有效管理异步任务成为关键。Kotlin协程提供了一种优雅的并发操作处理方式,使异步编程更简单直观。本文通过具体示例介绍如何使用Kotlin协程优化Android应用性能,包括网络数据加载和UI更新。首先需在`build.gradle`中添加coroutines依赖。接着,通过定义挂起函数执行网络请求,并在`ViewModel`中使用`viewModelScope`启动协程,结合`Dispatchers.Main`更新UI,避免内存泄漏。使用协程不仅简化代码,还提升了程序健壮性。
179 1
|
7月前
|
安全 Android开发 Kotlin
Android经典面试题之Kotlin延迟初始化的by lazy和lateinit有什么区别?
**Kotlin中的`by lazy`和`lateinit`都是延迟初始化技术。`by lazy`用于只读属性,线程安全,首次访问时初始化;`lateinit`用于可变属性,需手动初始化,非线程安全。`by lazy`支持线程安全模式选择,而`lateinit`适用于构造函数后初始化。选择依赖于属性特性和使用场景。**
205 5
Android经典面试题之Kotlin延迟初始化的by lazy和lateinit有什么区别?

热门文章

最新文章

  • 1
    如何修复 Android 和 Windows 不支持视频编解码器的问题?
  • 2
    【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
  • 3
    当flutter react native 等混开框架-并且用vscode-idea等编译器无法打包apk,打包安卓不成功怎么办-直接用android studio如何打包安卓apk -重要-优雅草卓伊凡
  • 4
    【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
  • 5
    APP-国内主流安卓商店-应用市场-鸿蒙商店上架之必备前提·全国公安安全信息评估报告如何申请-需要安全评估报告的资料是哪些-优雅草卓伊凡全程操作
  • 6
    【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
  • 7
    【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
  • 8
    【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
  • 9
    如何利用Java和Kotlin实现动态网页内容抓取
  • 10
    美团面试:MySQL为什么 不用 Docker部署?