Kotlin多线程

目录

线程的使用

线程的创建

例一:创建线程并输出Hello World

Thread对象的用法

start()

join()

interrupt()

线程安全

原子性

可见性

有序性

线程锁

ReentrantLock

ReadWriteLock


线程的使用

Java虚拟机中的多线程可以1:1映射至CPU中,即一个CPU线程跑一个任务,这叫并行,也可以N:1地运行,即一个CPU线程交替跑多个任务,看起来是同时地。这两种方法都叫并发

线程的创建

kotlin中,可以通过kotlin.concurrent包下的thread函数创建一个线程:

fun thread(start: Boolean = true,isDaemon: Boolean = false,contextClassLoader: ClassLoader? = null,name: String? = null,priority: Int = -1,block: () -> Unit
): Thread

该函数接收6个参数,必须定义block参数,因为它是线程的执行函数:

  • start: 如果为真,则立即执行
  • isDaemon: 如果为真,则会创建守护线程。当所有正在运行的线程都是守护线程时,Java虚拟机将自动退出
  • contextClassLoader: 线程中所使用的类加载器,又叫上下文类加载器。如果不指定类加载器,则会使用系统的类加载器
  • name: 线程的名字
  • priority: 线程的优先级。只有该参数大于0时才有效。线程的优先级在1-10之间,默认为5. 线程的优先级的最大值、最小值、默认值被分别定义在java.long包下Thread类的静态变量MAX_PRIORITY、MIN_PRIORITY和NORM_PRIORITY内
  • block: 一个回调函数,无参数,无返回值,线程运行调用此方法

该函数返回一个java.long包下的Thread对象,表示创建的线程。

因为该函数的前五个参数都有默认值,因此可以使用kotlin的语法糖,简化thread的用法:

thread { println("Hello World")
}

例一:创建线程并输出Hello World

import kotlin.concurrent.threadfun main() {thread {println("Hello World")}}

这就是这个例子的全部代码了,是不是非常简单?

在main方法里,创建了一个线程,线程执行时打印Hello World.

Thread对象的用法

我们提到,thread函数会返回一个Thread对象,那么,如何使用这个Thread对象呢?

首先,Thread类是用Java写的,所以它的函数原型是Java形式的

start()

Thread对象中有start()方法,表示执行线程:

public void start()

如果我们在thread方法中设置start参数为false,那么我们可以通过调用start()方法执行线程:

import kotlin.concurrent.threadfun main() {val th = thread(start = false) {println("Hello World")}println("准备启动线程")th.start()}

执行结果:

准备启动线程
Hello World
join()

join()方法等待线程执行结束:

public final void join()throws InterruptedException

如我们可以这样使用:

import kotlin.concurrent.threadfun main() {val th = thread {Thread.sleep(1000)println("th执行完成")}th.join()println("main执行完成")
}

 执行结果如下:

th执行完成
main执行完成

因此,join成功是main线程等待th线程结束

如果我们去掉th.join(),则输出:

main执行完成
th执行完成

这就是join()的基本用法

另外,如果当前线程(调用join()方法的线程)被任何线程中断,则抛出InterruptedException

异常,并不再等待:

import kotlin.concurrent.threadfun main() {val th = thread {val th2 = thread {Thread.sleep(1000)println("th2执行完成")}try {th2.join()}catch (e: InterruptedException){println("中断")}println("th执行完成")}th.interrupt()}

 执行结果:

中断
th执行完成
th2执行完成

因为main线程创建了th线程,th线程又创建了th2线程。th线程调用join()方法等待th2线程时,main线程中断了th线程,因此th线程中的join()方法停止等待,执行完成。之后,th2线程才执行完成

interrupt()

interrupt()中断线程。调用该方法时,将会把指定线程的Thread.interrupted()方法的返回值设为true,因此,要中断线程需要检测这个值。

public void interrupt()

 其用法如下:

import kotlin.concurrent.threadfun main() {val th = thread {while (true){if (Thread.interrupted()) break}println("th被中断")}Thread.sleep(1000)println("准备中断线程")th.interrupt()}

输出:

准备中断线程
th被中断

线程安全

线程安全必须同时满足原子性、可见性和有序性:

原子性

考虑这么一个代码:

import kotlin.concurrent.threadfun main() {var tmp = 0val th1 = thread {Thread.sleep(200)tmp++}val th2 = thread {Thread.sleep(200)tmp++}th1.join()th2.join()println(tmp)}

其中,tmp被增加了2次,因此应该返回2,可是我的输出结果为:

1

这是为什么呢?

我们知道,自增语句分三步:读取、增加、写入。在两个线程同时执行的时候,可能会出现类似以下情况:

时间第一个线程第二个线程
1读取tmp变量(0)
2计算tmp+1的值
3读取tmp变量(0)
4写入tmp+1的值到tmp变量(1)
5计算tmp+1的值
6写入tmp+1的值到tmp变量(1)

因此,由于线程之间并发运行,最终tmp的值为1。

之所以自增语句会出现这样的问题,是因为自增语句需要3块时间才能完成,不能一口气直接完成。如果自增可以直接完成,在非并行的情况下,就会出现以下情况:

时间第一个线程第二个线程
1tmp自增
2tmp自增

这样就不会有冲突了。

我们称这种直接完成而不被其他线程打断的操作叫原子操作,在kotlin中可以通过java.util.concurrent.atomic定义的支持原子操作的类,实现原子操作:

import java.util.concurrent.atomic.AtomicInteger
import kotlin.concurrent.threadfun main() {val tmp = AtomicInteger(0)val th1 = thread {Thread.sleep(200)tmp.incrementAndGet()}val th2 = thread {Thread.sleep(200)tmp.incrementAndGet()}th1.join()th2.join()println(tmp.get())}

注意:原子操作不适合并行时的问题,但由于现代电脑CPU少线程多的现状,大部分的情况都可以使用原子操作:

一个12核CPU有将近4000个线程

可见性

由于现代设备的线程有自己的缓存,有些时候当一个变量被修改后,其他线程可能看不到修改的信息,因此就会产生线程安全问题:

import kotlin.concurrent.threadfun main() {var boolean = trueval th1 = thread {Thread.sleep(200)boolean = falseprintln("已经将boolean设为false")}val th2 = thread {println("等待boolean为false")while (boolean){}}th1.join()th2.join()println("线程执行完毕")}

执行结果:

等待boolean为false
已经将boolean设为false
(无限循环)

 这是因为,当boolean被修改时,th2不能及时获得boolean的变化,所以跳不出循环,出现了可见性问题。我们可以通过Thread.yield()方法同步变量在线程内和进程内的数据:

import kotlin.concurrent.threadfun main() {var boolean = trueval th1 = thread {Thread.sleep(200)boolean = falseprintln("已经将boolean设为false")}val th2 = thread {println("等待boolean为false")while (boolean){Thread.yield()}}th1.join()th2.join()println("线程执行完毕")}

执行结果:

等待boolean为false
已经将boolean设为false
线程执行完毕

注意,Thread.yield()方法的真实作用是告诉调度器当前线程愿意放弃对处理器的使用,直到处理器重新调用这个线程,可以用以下表格来说明:

因此,Thread.yield()方法就可以抽空在合适的时机同步变量的数据,实现线程的可见性。

我们前面举的变量自增的例子也有可能是因为线程的可见性问题导致的。

有序性

我们在写代码时,往往认为程序是按顺序运行的,其实并不是。如果前后两个指令没有任何关联,处理器可能会先运行写在后面的省时指令,后运行写在前面的费时指令,这样可以起到节省资源的效果。在单线程中,这没有问题,但在多线程中,就出现了问题:

import kotlin.concurrent.threadfun main() {var a = 0var b = 0var x = -1var y = -1var count = 0while (true) {a = 0b = 0x = -1y = -1val th1 = thread {b = 1x = areturn@thread}val th2 = thread {a = 1y = breturn@thread}th1.join()th2.join()count++if (x == 0 && y == 0){println("第$count 次,($x,$y)")break}}}

输出:

第100010 次,(0,0)

按照正常的逻辑,这个程序的运行过程应该是类似这样的:

时间第一个线程第二个线程
1b=1
2a=1
3x=a(1)
4y=b(1)

时间第一个线程第二个线程
1b=1
2x=a(0)
3a=1
4y=b(1)

时间第一个线程第二个线程
1a=1
2y=b(0)
3b=1
4x=a(1)

无论如何,x和y都不可能同时为0,可是为什么原程序中,x和y都为0呢?

只有一种可能,类似这样:

x和y的赋值语句被处理器提到了前面,因此出现了有序性的问题 

@Volatile注解可以保证指定变量的可见性和有序性:

import kotlin.concurrent.thread@Volatile
var a = 0@Volatile
var b = 0@Volatile
var x = -1@Volatile
var y = -1fun main() {var count = 0while (true) {a = 0b = 0x = -1y = -1val th1 = thread {b = 1x = areturn@thread}val th2 = thread {a = 1y = breturn@thread}th1.join()th2.join()count++if (x == 0 && y == 0) {println("第$count 次,($x,$y)")break}}}

运行结果:

(无限循环)

 可见,@Volatile注解保证了其有序性。这个注解保证可见性和有序性的原理如下:

  • 可见性:给变量上一个Load屏障,每次读取数据的时候被强制从进程中读取最新的数据;同时上一个Store屏障,强制使每次修改之后强制刷新进程中的数据
  • 有序性:通过禁止重排屏障禁止指令重排:
    • StoreStore屏障:禁止StoreStore屏障的前后Store写操作重排
    • LoadLoad屏障:禁止LoadLoad屏障的前后Load读操作进行重排
    • LoadStore屏障:禁止LoadStore屏障的前面Load读操作跟LoadStore屏障后面的Store写操作重排
    • StoreLoad屏障:禁止LoadStore屏障前面的Store写操作跟后面的Load/Store 读写操作重排

线程锁

我们可以通过线程锁保证线程安全:

如果在操作对象之前,线程先声明:“这个对象是我的”,当另一个线程也想操作这个对象时,发现已经有人声明过了,那么它就等待,直到那个发布声明的线程又发了一个“这个对象不是我的了”的声明。当然,这两个声明其实起到了一个锁的作用,当声明“这个对象是我的”时,对象就被上了锁,当声明“这个对象不是我的了”时,对象的锁就被解开了

当然,上锁和解锁这一过程都必须保证原子性、可见性和有序性

我们可以如下修改代码:

import kotlin.concurrent.threadclass MyMutex{private var mutex: Boolean = false@Synchronizedfun lock(){while (mutex){}  // 等待解锁mutex = true     // 上锁return}@Synchronizedfun unlock(){mutex = false    // 解锁return}}fun main() {var tmp = 0val mutex = MyMutex()val th1 = thread {Thread.sleep(200)mutex.lock()tmp++mutex.unlock()}val th2 = thread {Thread.sleep(200)mutex.lock()tmp++mutex.unlock()}th1.join()th2.join()println(tmp)}

这里面使用了@Synchronized注解,可以保证方法的原子性、可见性和有序性。在这里面保证了上锁和解锁的原子性、可见性和有序性。

@Synchronized注解是这么保证方法的原子性、可见性和有序性的:

  • 原子性:在方法执行前加锁,执行后解锁,这样一个方法同时只能有一个线程使用
  • 可见性:在方法执行时给方法内的每个变量上一个Load屏障,每次读取数据的时候被强制从进程中读取最新的数据;同时上一个Store屏障,强制使每次修改之后强制刷新进程中的数据
  • 有序性:通过禁止重排屏障禁止指令重排:
    • StoreStore屏障:禁止StoreStore屏障的前后Store写操作重排
    • LoadLoad屏障:禁止LoadLoad屏障的前后Load读操作进行重排
    • LoadStore屏障:禁止LoadStore屏障的前面Load读操作跟LoadStore屏障后面的Store写操作重排
    • StoreLoad屏障:禁止LoadStore屏障前面的Store写操作跟后面的Load/Store 读写操作重排

当然,我们上面的代码只是简单实现了一个线程锁,kotlin中可以使用自带的线程锁:

ReentrantLock

ReentrantLock是一个递归互斥,在Java中叫可重入锁,允许同一个线程多次上锁,相应的,同一个线程上锁多少次,就要解锁多少次。为什么要允许线程多次上锁呢?

我们来看以下代码:

import kotlin.concurrent.threadclass MyMutex{private var mutex: Boolean = false@Synchronizedfun lock(){while (mutex){}  // 等待解锁mutex = true     // 上锁return}@Synchronizedfun unlock(){mutex = false    // 解锁return}}fun main() {val mutex1 = MyMutex()val mutex2 = MyMutex()val th1 = thread {mutex1.lock()println("th1 locked mutex1")     // 模拟操作受mutex1保护的资源Thread.sleep(200)mutex2.lock()println("th1 locked mutex2")      // 模拟操作受mutex2保护的资源mutex1.unlock()println("th1 unlocked mutex1")Thread.sleep(200)mutex2.unlock()println("th2 unlocked mutex2")return@thread}val th2 = thread {mutex2.lock()println("th1 locked mutex2")      // 模拟操作受mutex2保护的资源Thread.sleep(200)mutex1.lock()println("th1 locked mutex1")      // 模拟操作受mutex1保护的资源mutex2.unlock()println("th1 unlocked mutex2")Thread.sleep(200)mutex1.unlock()println("th2 unlocked mutex1")return@thread}th1.join()th2.join()println("线程执行完毕")}

运行结果:

th1 locked mutex1
th1 locked mutex2
(无限循环)

为什么会无限循环呢?因为th1锁定mutex1后想要锁定mutex2,却发现mutex2被th2锁定;而th2锁定mutex2后想要锁定mutex1,却发现mutex1被th1锁定,因此出现了无限循环的问题。我们称这种问题为死锁

使用递归互斥可以有效避免死锁问题:

import java.util.concurrent.locks.ReentrantLock
import kotlin.concurrent.threadfun main() {val mutex = ReentrantLock()val th1 = thread {mutex.lock()println("th1 locked mutex1")Thread.sleep(200)            // 模拟操作受mutex1保护的资源mutex.lock()println("th1 locked mutex2")      // 模拟操作受mutex2保护的资源mutex.unlock()println("th1 unlocked mutex1")Thread.sleep(200)mutex.unlock()println("th2 unlocked mutex2")return@thread}val th2 = thread {mutex.lock()println("th1 locked mutex2")      // 模拟操作受mutex2保护的资源Thread.sleep(200)mutex.lock()println("th1 locked mutex1")      // 模拟操作受mutex1保护的资源mutex.unlock()println("th1 unlocked mutex2")Thread.sleep(200)mutex.unlock()println("th2 unlocked mutex1")return@thread}th1.join()th2.join()println("线程执行完毕")}

其中,代码

val mutex = ReentrantLock()

表示创建一个可重入锁,这个对象的lock()和unlock()方法分别表示上锁和解锁

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/263855.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

在Node.js中如何实现用户身份验证和授权

当涉及到构建安全的应用程序时,用户身份验证和授权是至关重要的一环。在Node.js中,我们可以利用一些流行的库和技术来实现这些功能,确保我们的应用程序具有所需的安全性。本篇博客将介绍如何在Node.js中实现用户身份验证和授权。 用户身份验…

留存测试数据,Apipost接口用例详解

接口用例可以在不影响源接口数据的情况下对接口添加多个用例,方便测试并保存测试数据。 创建用例 左侧目录选择接口后进入接口用例页面,点击添加用例 在弹出窗口中修改各种参数。如登录接口,可修改用户名为空,并添加断言。 执行…

图解KMP算法

目录 1.最长公共前后缀1.1前缀1.2后缀1.3最长公共前后缀 2、KMP算法过程2.1例子12.2例子22.3Python代码:2.4next数组的计算过程 1.最长公共前后缀 1.1前缀 前缀说的是一个字符串除了最后一个字符以外,所有的子串都算是前缀。 前缀字符串:A…

Apache celeborn 安装及使用教程

1.下载安装包 https://celeborn.apache.org/download/ 测0.4.0时出现https://github.com/apache/incubator-celeborn/issues/835 2.解压 tar -xzvf apache-celeborn-0.3.2-incubating-bin.tgz 3.修改配置文件 cp celeborn-env.sh.template celeborn-env.shcp log4j2.xml.…

Dear ImGui的UE5.3集成实践

Dear ImGui一直较为火热,这是一个调试使用并且可以响应快速迭代的Gui库,甚至可以做到在任何代码块中调用API即显示。如果你想更多的了解一下可访问其官方网站:https://www.dearimgui.org/ 那么本文就来在UE5中尝试踩坑使用它。 UE4.26版本 …

数据可视化基础与应用-01-数据可视化概述

总结 本系列是数据可视化基础与应用的第02篇,主要介绍数据可视化概述,包括数据可视化的历史,原理,工具等。 认识大数据可视化 数据是什么 信息科学领域面临的一个巨大挑战是数据爆炸。据IDC Global DataSphere统计&#xff0c…

EXCEL 在列不同单元格之间插入N个空行

1、第一步数据,要求在每个数字之间之间插入3个空格 2、拿数据个数*(要插入空格数1) 19*4 3、填充 4、复制数据到D列 5、下拉数据,选择复制填充这样1-19就会重复4次 6、全选数据D列排序,这样即完成了插入空格 以…

贪心算法---前端问题

1、贪心算法—只关注于当前阶段的局部最优解,希望通过一系列的局部最优解来推出全局最优----但是有的时候每个阶段的局部最优之和并不是全局最优 例如假设你需要找给客户 n 元钱的零钱,而你手上只有若干种面额的硬币,如 1 元、5 元、10 元、50 元和 100…

matlab滤波器设计

1、内容简介 略 51-可以交流、咨询、答疑 2、内容说明 略 3、仿真分析 略 matlab滤波器设计-butter、ellip、cheby1、cheby2_哔哩哔哩_bilibili 4、参考论文 略

「C#」WPF学习笔记-基础类及继承关系

1、DependencyObject DependencyObject是WPF中依赖属性系统的核心,它为WPF的数据绑定、动画和属性共享等功能提供了支持,是一个非常重要的基类。 其主要特点和职责包括: 依赖属性系统:DependencyObject 是所有支持依赖属性的类…

Python和Jupyter简介

在本notebook中,你将: 1、学习如何使用一个Jupyter notebook 2、快速学习Python语法和科学库 3、学习一些IPython特性,我们将在之后教程中使用。 这是什么? 这是只为你运行在一个个人"容器"中的一个Jupyter noteboo…

基于FPGA的I2C接口控制器(包含单字节和多字节读写)

1、概括 前文对IIC的时序做了详细的讲解,还有不懂的可以获取TI的IIC数据手册查看原理。通过手册需要知道的是IIC读、写数据都是以字节为单位,每次操作后接收方都需要进行应答。主机向从机写入数据后,从机接收数据,需要把总线拉低来…

网络安全“三保一评”深度解析

“没有网络安全就没有国家安全”。近几年,我国法律法规陆续发布实施,为承载我国国计民生的重要网络信息系统的安全提供了法律保障,正在实施的“3保1评”为我国重要网络信息系统的安全构筑了四道防线。 什么是“3保1评”? 等保、分…

QlikSense CyberSecurity : Configuring preferred Cipher Suites

You can rank the preferred cipher suites that Qlik License Service uses to encrypt and decrypt the signed key license.您可以对Qlik许可证服务用于加密和解密签名密钥许可证的首选密码套件进行排序。 The Qlik License Service is included in Qlik Sense Enterprise …

react中修改state中的值无效?

// 初始化state state {personArr:[{name:张三,id:1},{name:李四,id:2},{name:王五,id:3}] }componentDidMount(){const newName 赵六const indexUpdate 1const newArr this.state.personArr.map((item,index)>{if(indexUpdate index){return {...item,name:newName}}e…

【C++ QT项目5】——基于HTTP与JSON数据流的天气预报界面设计

【C QT项目5】——基于HTTP与JSON数据流的天气预报界面设计 一、项目概述二、UI设计与stylesheet样式表三、天气预报数据接口四、JSON数据4.1 概述4.2 QT生成JSON数据4.3 QT解析JSON数据4.4 将JSON数据解析到QMap中 五、软件开发网络通信架构5.1 BS架构/CS架构5.2 HTTP基本概念…

leetcode hot100 买卖股票的最佳时机二

注意,本题是针对股票可以进行多次交易,但是下次买入的时候必须保证上次买入的已经卖出才可以。 动态规划可以解决整个股票买卖系列问题。 dp数组含义: dp[i][0]表示第i天不持有股票的最大现金 dp[i][1]表示第i天持有股票的最大现金 递归公…

SpringBoot Admin 详解

SpringBoot Admin 详解 一、Actuator 详解1.Actuator原生端点1.1 监控检查端点:health1.2 应用信息端点:info1.3 http调用记录端点:httptrace1.4 堆栈信息端点:heapdump1.5 线程信息端点:threaddump1.6 获取全量Bean的…

栈的最后表演:逆波兰表达式求值

前言 今天刷题遇到了逆波兰表达式,死亡的记忆突然开始攻击我,好嘛,既然根基不牢,那么就一次性给他搞明白了! 一、算术表达式求值 算术表达式又叫中缀表达式,如果直接给出一个中缀表达式让我们求值&#…

Spring Boot 笔记 025 主界面

1.1 路由搭建 1.1.1 安装vue router npm install vue-router4 1.1.2 在src/router/index.js中创建路由器,并导出 import { createRouter, createWebHistory } from vue-router//导入组件 import LoginVue from /views/Login.vue import LayoutVue from /views/La…