Kotlin:该如何实现多线程同步?

本文详细介绍了Java中的多种并发控制机制,包括Synchronized、ReentrantLock、CountDownLatch、CyclicBarrier和CAS,以及Future和CompletableFuture在处理异步任务中的应用,特别强调了在Android开发中提升技能的重要性。
摘要由CSDN通过智能技术生成

val t1 = Thread { s1 = task1() }

val t2 = Thread { s2 = task2() }

t1.start()

t2.start()

t1.join()

t2.join()

task3(s1, s2)

}

方式2:线程锁

主要包括:Synchronized、ReentrantLock、CountDownLatch、CyclicBarrier

Synchronized

@Test

fun test_synchrnoized() {

lateinit var s1: String

lateinit var s2: String

Thread {

synchronized(Unit) {

s1 = task1()

}

}.start()

s2 = task2()

synchronized(Unit) {

task3(s1, s2)

}

}

这里需要特别注意的是:为了同步多个并行任务的结果则需要声明n个锁, 即需嵌套n个 synchronized

ReentrantLock

相对于Synchronized,ReentrantLock的使用则不会出现嵌套 synchrnoized 的问题,但仍需创建多个 lock 从而管理多个不同的线程任务。

fun test_ReentrantLock() {

lateinit var s1: String

lateinit var s2: String

val lock = ReentrantLock()

Thread {

lock.lock()

s1 = task1()

lock.unlock()

}.start()

s2 = task2()

lock.lock()

task3(s1, s2)

lock.unlock()

}

这里需要额外说明的是,阻塞队列BlockingQueue内部是通过ReentrantLock实现的,所以其也能实现线程同步,但其应用场景是:生产/消费场景中的同步

fun test_blockingQueue() {

lateinit var s1: String

lateinit var s2: String

val queue = SynchronousQueue()

Thread {

s1 = task1()

queue.put(Unit)

}.start()

s2 = task2()

queue.take()

task3(s1, s2)

}

CountDownLatch

JUC 中的锁大都基于 AQS 实现的,可以分为独享锁和共享锁。ReentrantLock 就是一种独享锁。相比之下,共享锁更适合本场景,不需为了每个任务都创建单独的锁。

@Test

fun test_countdownlatch() {

lateinit var s1: String

lateinit var s2: String

val cd = CountDownLatch(2)

Thread() {

s1 = task1()

cd.countDown()

}.start()

Thread() {

s2 = task2()

cd.countDown()

}.start()

cd.await()

task3(s1, s2)

}

CyclicBarrier

原理:让一组线程到达一个同步点后再一起继续运行,其中任意一个线程未达到同步点,其他已到达的线程均会被阻塞。

@Test

fun test_CyclicBarrier() {

lateinit var s1: String

lateinit var s2: String

val cb = CyclicBarrier(3)

Thread {

s1 = task1()

cb.await()

}.start()

Thread() {

s2 = task1()

cb.await()

}.start()

cb.await()

task3(s1, s2)

}

需要特别注意的是:与 CountDownLatch 的区别在于 CountDownLatch 是一次性的,而 CyclicBarrier 可以被重置后循环利用

方式3:CAS

原理:基于 CAS 的原子类计数

应用场景:一些cpu密集型的短任务同步(因为会比较损耗资源)

fun test_cas() {

lateinit var s1: String

lateinit var s2: String

val cas = AtomicInteger(2)

Thread {

s1 = task1()

cas.getAndDecrement()

}.start()

Thread {

s2 = task2()

cas.getAndDecrement()

}.start()

while (cas.get() != 0) {}

task3(s1, s2)

}

这里需要特别说明的是,看到 CAS 的无锁实现,很多人会想到 volatile:并非线程安全,因为volatile 能保证可见性,但是不能保证原子性,cnt-- 并非线程安全,需要加锁操作

fun test_Volatile() {

lateinit var s1: String

lateinit var s2: String

Thread {

s1 = task1()

cnt–

}.start()

Thread {

s2 = task2()

cnt–

}.start()

while (cnt != 0) {

}

task3(s1, s2)

}

方式4:Future

Java 1.5 开始提供了一种可以在任务执行结束时返回结果的线程同步方式:Callable 和 Future 。即不需通过定义变量来记录结果了。

// 通过 future.get(),可以同步等待结果返回,写起来非常方便

fun test_future() {

val future1 = FutureTask(Callable(task1))

val future2 = FutureTask(Callable(task2))

Executors.newCachedThreadPool().execute(future1)

Executors.newCachedThreadPool().execute(future2)

task3(future1.get(), future2.get())

}

这里需要特别说明的是,future.get() 虽然方便,但是会阻塞线程。所以在 Java 8 中引入了 CompletableFuture :他实现了 Future 接口的同时实现了 CompletionStage 接口,即可针对多个 CompletionStage 进行逻辑组合、实现复杂的异步编程。以回调的形式避免了线程阻塞

fun test_CompletableFuture() {

CompletableFuture.supplyAsync(task1)
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

结语

看到这篇文章的人不知道有多少是和我一样的Android程序员。

35岁,这是我们这个行业普遍的失业高发阶段,这种情况下如果还不提升自己的技能,进阶发展,我想,很可能就是本行业的职业生涯的终点了。

我们要有危机意识,切莫等到一切都成定局时才开始追悔莫及。只要有规划的,有系统地学习,进阶提升自己并不难,给自己多充一点电,你才能走的更远。

千里之行始于足下。这是上小学时,那种一元钱一个的日记本上每一页下面都印刷有的一句话,当时只觉得这句话很短,后来渐渐长大才慢慢明白这句话的真正的含义。

有了学习的想法就赶快行动起来吧,不要被其他的事情牵绊住了前行的脚步。不要等到裁员时才开始担忧,不要等到面试前一晚才开始紧张,不要等到35岁甚至更晚才开始想起来要学习要进阶。

给大家一份系统的Android学习进阶资料,希望这份资料可以给大家提供帮助。

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

还不提升自己的技能,进阶发展,我想,很可能就是本行业的职业生涯的终点了。

我们要有危机意识,切莫等到一切都成定局时才开始追悔莫及。只要有规划的,有系统地学习,进阶提升自己并不难,给自己多充一点电,你才能走的更远。

千里之行始于足下。这是上小学时,那种一元钱一个的日记本上每一页下面都印刷有的一句话,当时只觉得这句话很短,后来渐渐长大才慢慢明白这句话的真正的含义。

有了学习的想法就赶快行动起来吧,不要被其他的事情牵绊住了前行的脚步。不要等到裁员时才开始担忧,不要等到面试前一晚才开始紧张,不要等到35岁甚至更晚才开始想起来要学习要进阶。

给大家一份系统的Android学习进阶资料,希望这份资料可以给大家提供帮助。
[外链图片转存中…(img-aM0NpLTw-1713391117626)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 7
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值