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查看更多精彩文章!

目录
相关文章
|
17天前
|
安全 Go
Golang语言goroutine协程并发安全及锁机制
这篇文章是关于Go语言中多协程操作同一数据问题、互斥锁Mutex和读写互斥锁RWMutex的详细介绍及使用案例,涵盖了如何使用这些同步原语来解决并发访问共享资源时的数据安全问题。
38 4
|
4天前
|
Android开发 Kotlin
Android经典面试题之Kotlin的==和===有什么区别?
本文介绍了 Kotlin 中 `==` 和 `===` 操作符的区别:`==` 用于比较值是否相等,而 `===` 用于检查对象身份。对于基本类型,两者行为相似;对于对象引用,`==` 比较值相等性,`===` 检查引用是否指向同一实例。此外,还列举了其他常用比较操作符及其应用场景。
160 93
|
1月前
|
并行计算 数据挖掘 大数据
[go 面试] 并行与并发的区别及应用场景解析
[go 面试] 并行与并发的区别及应用场景解析
|
1月前
|
Java 程序员 调度
面试准备-并发
面试准备-并发
|
1月前
|
消息中间件 Java 中间件
复盘女朋友面试4个月的并发面试题
该文章主要复盘了关于并发的面试题,包括线程池的使用场景、原理、参数合理化设置,以及ThreadLocal、volatile、synchronized关键字的使用场景和原理,还介绍了juc并发工具包中aqs的原理,强调在面试中要将自己理解的点与面试官讲透。
复盘女朋友面试4个月的并发面试题
|
1月前
|
JavaScript 前端开发 Java
面试官:假如有几十个请求,如何去控制并发?
面试官:假如有几十个请求,如何去控制并发?
|
1月前
|
安全 Go 调度
[go 面试] 深入理解并发控制:掌握锁的精髓
[go 面试] 深入理解并发控制:掌握锁的精髓
|
1月前
|
算法 Go 数据库
[go 面试] 并发与数据一致性:事务的保障
[go 面试] 并发与数据一致性:事务的保障
|
2月前
|
安全 Android开发 Kotlin
Android经典面试题之Kotlin中常见作用域函数
**Kotlin作用域函数概览**: `let`, `run`, `with`, `apply`, `also`. `let`安全调用并返回结果; `run`在上下文中执行代码并返回结果; `with`执行代码块,返回结果; `apply`配置对象后返回自身; `also`附加操作后返回自身
40 8
|
2月前
|
Android开发 开发者
Android经典面试题之SurfaceView和TextureView有什么区别?
分享了`SurfaceView`和`TextureView`在Android中的角色。`SurfaceView`适于视频/游戏,独立窗口低延迟,但变换受限;`TextureView`支持复杂变换,视图层级中渲染,适合动画/视频特效,但性能略低。两者在性能、变换、使用和层级上有差异,开发者需按需选择。
45 1