Java线程和Go协程

Java线程和Go协程

Java线程和Go协程都是用于并发编程的工具,但在实现和使用上有一些不同。

Java线程模型

Java线程是Java语言提供的一种并发编程的机制,它允许程序在同一时间执行多个任务。Java线程是基于操作系统的线程实现的,每个线程都有自己的堆栈和程序计数器,并且可以通过调度器进行调度。Java线程可以通过继承Thread类或实现Runnable接口来创建和启动。

两种方式启动线程:

  • 继承Thread。

  • 实现Runnable接口。

class MyThread extends Thread {public void run() {// 线程的主要逻辑// ...}
}public class Main {public static void main(String[] args) {MyThread thread = new MyThread();thread.start(); // 启动线程}
}
class MyRunnable implements Runnable {public void run() {// 线程的主要逻辑// ...}
}public class Main {public static void main(String[] args) {MyRunnable myRunnable = new MyRunnable();Thread thread = new Thread(myRunnable);thread.start(); // 启动线程}
}

Java同步机制

  1. synchronized关键字: Java提供了synchronized关键字来控制对共享资源的并发访问,确保线程安全。

  2. ReentrantLock: ReentrantLock是一种基于AQS框架的可重入互斥锁,提供了比synchronized更多的功能。

AQS原理参考之前文章:《Java AQS实现》。


Java并发实现是基于操作系统调度的,即程序负责创建线程,操作系统负责调度。

这种方式有两大不足:

  1. 复杂性:

    • 创建容易、退出难。

    • 需要父线程去通知并等待子线程退出(join)。

    • 并发单元间通信困难:涉及到共享内存,就会用到锁,容易出现死锁。

    • 线程栈大小的设置。

  2. 扩展性:

    • 不能创建大量线程,因为每个线程占用的资源不小,操作系统调度切换线程的代价也不小。

    • 对于很多网络服务程序,由于不能大量创建线程,就要在少量线程里做网络的多路复用,即使用epoll/kqueue。

Go协程模型

Go协程是Go语言提供的一种轻量级的并发编程机制。

Go协程是由Go语言运行时环境管理的,它可以在同一个线程上运行多个协程。Go协程使用关键字"go"来创建,可以通过go关键字将一个函数调用转换为一个协程。Go协程之间通过通道(channel)进行通信,以实现数据的传递和同步。

Go中启动协程:

func main() {// 启动一个协程go sayHello("Hello from Goroutine 1")
}

Go同步机制:

  • Channel: Channel是Go中用于协程间通信和同步的主要机制。

  • Mutex: Mutex用于控制共享资源的访问,保证数据的完整性和一致性。

Go始终推荐以CSP(通信进程顺序)模型风格构建并发程序,也就是使用channel。channel实现了CSP模型中的输入/输出原语。

Mutex示例:

var (counter = 0mutex   sync.Mutex
)func incrementCounter(wg *sync.WaitGroup) {defer wg.Done()// 加锁mutex.Lock()defer mutex.Unlock()// 访问共享变量counter++fmt.Printf("Counter: %d\n", counter)
}func main() {var wg sync.WaitGroupfor i := 0; i < 5; i++ {wg.Add(1)go incrementCounter(&wg)}// 等待所有 Goroutines 完成wg.Wait()
}

Channel示例:

var (counter = 0
)func incrementCounter(wg *sync.WaitGroup, ch chan bool) {defer wg.Done()// 发送消息通知其他 Goroutine 进行更新ch <- true// 访问共享变量counter++fmt.Printf("Counter: %d\n", counter)// 发送消息通知完成<-ch
}func main() {var wg sync.WaitGroupch := make(chan bool, 1)for i := 0; i < 5; i++ {wg.Add(1)go incrementCounter(&wg, ch)}// 等待所有 Goroutines 完成wg.Wait()// 关闭通道close(ch)
}

Goroutine调度器 GPM 模型:

  • G(Goroutine):协程(轻量级线程)。存储了 goroutine 的执行栈信息、goroutine状态及goroutine的任务函数等。

  • P(Processor):包含了运行 goroutine 的资源,如果线程想运行 goroutine,必须先获取P,P中还包含了可运行的G队列。

  • M(Thread):代表真正的执行计算资源。从P的本地队列中获取G,切换到G的执行栈上并执行G的函数。

在go中,线程 M 是运行 goroutine 的实体,调度器 P 的功能是把可运行的 goroutine 分配到工作线程上。

主要流程:

  1. go func来创建一个goroutine。

  2. 有两个存储G的队列,一个是局部调度器P的本地队列,一个是全局队列。先创建的G会先保存在P的本地队列,本地满了后会保存在全局队列中。

  3. G只能运行在M中,一个M必须持有一个P,M和P是1:1的关系。M会从P的本地队列弹出一个可执行状态的G来执行,如果P的本地队列为空,就会从其它线程的P队列中偷取G执行。

  4. 当M执行某一个G时发生了syscall或者阻塞操作,M会阻塞,如果当前有一些G在执行,runtime会把这个线程M从P中摘除,然后再创建一个新的操作系统的线程(如果有空闲的线程可用就复用空闲线程)来服务于这个P。

  5. 当M系统调用结束时候,这个G会尝试获取一个空闲的P执行,并放入到这个P的本地队列。如果获取不到P,那么这个线程M变成休眠状态,加入到空闲线程中,然后这个G会被放入全局队列中。

Go协程调度策略

源码:go1.20/src/runtime/proc.go

//找到一个可运行的协程以执行
func findRunnable() (gp *g, inheritTime, tryWakeP bool) {// 偶尔从全局队列拿取协程保证公平性if pp.schedtick%61 == 0 && sched.runqsize > 0 {...}// 本地队列获取协程if gp, inheritTime := runqget(pp); gp != nil {return gp, inheritTime, false}// 全局队列获取协程if sched.runqsize != 0 {...}// 从网络轮询队列中获取事件//当一个goroutine(G、协程)在等待网络响应或某种I/O操作时,它会被阻塞,此时不会影响P队列中的其它G执行。//阻塞的goroutine等待网络响应数据到达后,G1会被从新放入P的本地队列,本地队列满了会被放入全局队列等待调度。//如果G被阻塞在某个channel操作或者网络I/O操作上,那么G会被放置到某个等待队列中,而M会尝试运行P的下一个可运行的G、if netpollinited() && netpollWaiters.Load() > 0 && sched.lastpoll.Load() != 0 {...}// 从其它P中窃取一半的任务,窃取个数n = x - x / 2。假如x有3个元素,则窃取2个if mp.spinning || 2*sched.nmspinning.Load() < gomaxprocs-sched.npidle.Load() {...}}

按如下顺序调度获取协程执行:

  1. 1/61的概率从全局队列取协程;

  2. 从本地队列获取协程;

  3. 从全局队列获取协程;

  4. 从网络轮询队列中获取协程事件;

  5. 从其它P队列中窃取协程。


阻塞调度:

  • 网络I/O阻塞:
    当一个 goroutine 在等待网络响应或某种I/O操作时,它会被阻塞,此时G会被放置到某个等待队列中,而M会尝试运行P的下一个可运行G。也就是说P中其它的G仍然可以并发执行。

    待阻塞的goroutine等待网络响应数据到达后,G会被重新放入P的本地队列,本地队列满了会被放入全局队列等待调度。

  • 系统调用阻塞:
    当一个G被阻塞在系统调用上,那么G会阻塞,执行该G的M也会解绑P,与G一起进行入阻塞状态。此时有空闲的M,则与P绑定并执行P中其它的G,没有空闲的M则创建新的M绑定P并执行。

    当系统调用返回后,阻塞的G会尝试获取一个可用的P,有可用的P则将之前运行G的M与P绑定继续运行G。没有可用的P则G与M的关联解除,并将G放入全局队列等待调度。


窃取:

从其它P中窃取一半的任务,窃取个数n = x - x / 2。假如x有3个元素,则窃取2个。

Go 中协程窃取机制和 Java 并行流中窃取任务机制思想一致,都是从其它队列偷取任务,放到自个队列中执行。

Java并行流任务窃取可见《ForkJoinPool源码解析》。

总结

Java中需要手动管理线程的生命周期和同步机制。为了复用线程,还需要创建线程池来达到线程的复用,此时线程池的参数也需要用户自己调试。

而在Go中是由Go 语言自动管理协程的生命周期和调度。此外,Go协程的通信机制更加灵活,可以通过通道实现协程之间的数据传递和同步。

参考资料

  1. 《Go语言精进之路》。

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

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

相关文章

【校招VIP】测试计划之hashmap分析

考点介绍&#xff1a; HashMap是Java程序员使用频率最高的用于映射键值对(key和value)处理的数据类型。随着JDK版本的跟新&#xff0c;JDK1.8对HashMap底层的实现进行了优化&#xff0c;列入引入红黑树的数据结构和扩容的优化等。 测试计划之hashmap分析-相关题目及解析内容可…

单片机采集传感器数据(整形,浮点型)modbus上传

浮点型数据 占两个寄存器&#xff08;四个字节&#xff09; short 整形 占一个寄存器 &#xff08;两个字节&#xff09; 注意&#xff01;&#xff01;&#xff01;&#xff01; stm32 是小端模式&#xff0c;而modbus解析数据是大端模式 所以先发送高字节 如int a16777220…

Web3 solidity编写cancelorder取消订单函数 并梳理讲述逻辑

上文 Web3 solidity订单池操作 中 我们讲述了订单池的基本概念 并手动编写了创建订单的操作 最近的 我们还是先将 ganache 环境起起来 然后 我们打开项目 上文中 我们写了makeOrder创建订单的函数 但是 也带出一个问题 我们创建之后 如果不要了 怎么干掉呀&#xff1f; js中我…

一键部署k8s集群

前置动作 关闭防火墙 systemctl disable firewalld && systemctl stop firewalld 关闭SELinux sed -i s#SELINUXenforcing#SELINUXdisabled#g /etc/selinux/config && grep SELINUXdisabled /etc/selinux/config setenforce 0 getenforce 关闭swap # 关闭…

centos密码过期导致navicat无法通过SSH登录阿里云RDS问题

具体错误提示&#xff1a;2013 - Lost connection to server at "hand hake: reading initial communication packet, system error: 0 解决办法&#xff1a;更新SSH服务器密码

docker 跨平台构建镜像

我们在开发环境构建的镜像在生产环境大多不可用&#xff0c;我们在开发中一般使用 Windows 或者 MAC 系统&#xff0c;部署多半是 linux 环境。那么这篇文章能帮到你。 文章目录 首先构建环境进阶 首先 首先你需要有一个 Dockerfile 文件。 举例&#xff1a;这里以一个 pytho…

MySQL——数据库以及数据表的创建

创建数据库 回到刚才创建数据库的问题&#xff0c;我们在创建数据库的时候可以通过添加一个参数&#xff0c;这个参数的意义在于当我们创建的数据库已经存在的时候则不会创建&#xff0c;也不会报错&#xff0c;如果不使用这个参数&#xff0c;则我们在重复创建一个已经存在的…

k8s集群中集群方式安装nacos

1、前提条件 一个k8s集群&#xff0c;其中有三个master 节点&#xff0c;这三个节点的标签名称为etcd 三个master节点的ip 分别为&#xff1a;192.165.187.170 、192.165.187.171、192.165.187.172一个mysql 数据库&#xff0c; 数据库的ip 为&#xff1a;192.165.187.180 用户…

python Playwright优化页面等待和处理异步操作

在使用 Playwright 进行页面自动化时&#xff0c;优化页面等待和处理异步操作是非常重要的&#xff0c;可以提高脚本的稳定性和执行效率。 优化页面等待和处理异步操作的建议 **1. 使用正确的等待条件&#xff1a;**Playwright 提供了多种等待条件&#xff0c;如等待元素出现…

grep 的非贪婪模式

实测grep的非贪婪模式是-P参数&#xff0c;加上匹配字符串.*带?&#xff08;.*?&#xff09;&#xff1a; #贪婪模式&#xff08;默认&#xff09; grep "Product.*“LME:AA ei6_gateway.log --color #非贪婪模式 grep -P "Product.*?“LME:AA ei6_gateway.log -…

机器学习笔记之最优化理论与方法(八)无约束优化问题——常用求解方法(中)

机器学习笔记之最优化理论与方法——基于无约束优化问题的常用求解方法[中] 引言回顾&#xff1a;最速下降算法的缺陷经典牛顿法基本介绍经典牛顿法的问题经典牛顿法的优点与缺陷经典牛顿法示例 修正牛顿法介绍拟牛顿法拟牛顿法的算法过程 矩阵 B k 1 \mathcal B_{k1} Bk1​的…

亚马逊鲲鹏AI智能养号好用吗?怎么使用的?

亚马逊鲲鹏AI智能一键养号可以根据AI功能页面的姓名、年龄、职业、爱好等生成一批不同的AI角色&#xff0c;账号绑定这些角色后就可以自动浏览进行养号了。 功能特点 1、自动生成AI姓名、随机选择角色性别、自由设置AI年龄 2、根据勾选的AI职业、AI爱好进行随机生成AI关键词进…