一文了解Go 并发与并行

 更新时间:2024年05月15日 10:12:18   作者:比猪聪明  
并发性和并行性是是两个既有联系又有所区别的概念,本文主要介绍了Go并发与并行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

Go语言,由Google的Robert Griesemer、Rob Pike和Ken Thompson于2009年开发,是一种静态类型、垃圾回收、多线程并发的编程语言。Go语言的设计目标是简单、高效、易于使用。它的并发模型非常强大,可以轻松地处理大量并发任务,这使得Go语言成为现代高性能并发应用的理想选择。

在Go语言(Golang)中,并发(Concurrency)和并行(Parallelism)是两个既有联系又有所区别的概念

并发(Concurrency)

并发指的是在一段时间内,多个任务能够开始、执行和完成,这些任务在宏观上看起来是同时进行的,尽管在微观层面上它们可能并不是真正的同时执行。Go语言通过goroutines(轻量级线程)和channels(用于goroutines间通信的安全通道)来支持高效率的并发编程。并发的重点在于任务的组织和管理,使得即使在单个CPU核心上,也能通过时间切片和任务调度,给予用户多个任务同时进行的错觉。

并行(Parallelism)

并行则是指在物理上或逻辑上的多个处理器(或多核CPU)同时执行多个任务。当程序能够充分利用多核处理器的多个核心,使得不同的任务或任务的部分确实同时在不同的处理器上执行,这就是并行。Go语言的运行时能够自动利用多核处理器,使得当有足够的goroutines需要运行时,它们可以被分配到不同的CPU核心上并行执行,从而提升程序的执行效率。

goroutine的实现原理

goroutine的实现原理主要依赖于Go语言的运行时(runtime)和调度器(scheduler)。当我们创建一个goroutine时,运行时会为其分配一个独立的栈空间,并将其加入到调度器的运行队列中。调度器会根据goroutine的优先级和状态(运行、休眠、阻塞等)来调度它们的执行。

channel的实现原理

channel的实现原理主要依赖于Go语言的运行时和内存模型。当我们创建一个channel时,运行时会为其分配一个缓冲区,用于存储数据。channel的读写操作是原子的,这意味着它们不会被中断。当一个goroutine向channel中写入数据时,运行时会将数据放入缓冲区,并通知其他goroutine。当一个goroutine从channel中读取数据时,运行时会从缓冲区中取出数据,并通知其他goroutine。

关系与区别

  • 区别:并发关注的是任务的执行方式和任务间的协作,能够在单核或多核环境下工作,不保证任务绝对同时执行。而并行关注的是任务的实际同时执行,需要多核环境来实现。
  • 联系:并发是并行的基础,没有并发就没有并行。并发使得程序能够在逻辑上分解为多个独立执行的单元,而并行则是在硬件层面实现这些单元的同时执行,从而达到更高的性能。 
for i := 1; i <= 10; i++ {

	go func() {
		fmt.Println("123")
	}()
}

 这段代码展示了Go语言中并发的特性。for循环与通过go关键字启动的goroutines之间是并发执行的。具体分析如下:

  • 并发性: 当for循环每次迭代时,它都会启动一个新的goroutine并通过go func() {...}()表达式。这意味着循环继续进行下一次迭代时,刚启动的goroutine几乎立刻开始执行,无需等待前一个goroutine完成。因此,这些goroutine的执行是并发的——它们的执行在时间上重叠,不过实际的执行顺序取决于Go运行时的调度策略和可用的处理器核心。

  • 并行性: 是否并行执行(即是否同时在多个处理器核心上执行)则取决于运行时环境和当前系统的负载。如果系统有多个处理器核心且goroutines的数量足够多以至于它们不能全部在一个核心上高效执行,Go的调度器可能会将这些goroutine分布在不同的核心上并行执行。然而,对于这段特定的代码,由于goroutine内的工作非常轻(仅打印一个空字符串),并行收益不大,甚至可能全部在一个核心上顺序快速执行完,因为启动goroutine和打印操作都非常快。

总结来说,这段代码展示了并发执行,而并行性取决于运行时的具体情况,但就这个简单的示例而言,并行效果可能不明显。

到此这篇关于一文了解Go 并发与并行的文章就介绍到这了,更多相关Go 并发与并行内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Go语言类型内嵌和结构体内嵌的具体使用

    Go语言类型内嵌和结构体内嵌的具体使用

    本文主要介绍了Go语言类型内嵌和结构体内嵌的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • 一篇文章带你轻松搞懂Golang的error处理

    一篇文章带你轻松搞懂Golang的error处理

    在进行后台开发的时候,错误处理是每个程序员都会遇到的问题,下面这篇文章主要给大家介绍了关于Golang中error处理的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-07-07
  • Go语言pointer及switch fallthrough实战详解

    Go语言pointer及switch fallthrough实战详解

    这篇文章主要为大家介绍了Go语言pointer及switch fallthrough实战详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • go defer避坑指南之拆解延迟语句

    go defer避坑指南之拆解延迟语句

    这篇文章主要为大家详细介绍了go defer避坑指南之如何拆解延迟语句,掌握正确使用方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-11-11
  • golang使用iconv报undefined:XXX的问题处理方案

    golang使用iconv报undefined:XXX的问题处理方案

    这篇文章主要介绍了golang使用iconv报undefined:XXX的问题处理方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-03-03
  • 解决GO编译时避免引入外部动态库的问题

    解决GO编译时避免引入外部动态库的问题

    最近碰到一个问题,有一个流量采集的组件中使用到了github.com/google/gopacket 这个库,这个库使用一切正常,但是唯独有一个缺点,编译后的二进制文件依赖于libpcap.so的动态库,这篇文章主要介绍了GO编译时避免引入外部动态库的解决方法,需要的朋友可以参考下
    2022-10-10
  • 如何在golang中使用shopspring/decimal来处理精度问题

    如何在golang中使用shopspring/decimal来处理精度问题

    本文主要介绍了如何在golang中使用shopspring/decimal来处理精度问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • Go语言转化php数组的示例代码

    Go语言转化php数组的示例代码

    这篇文章主要为大家详细介绍了Go语言如何实现转化php数组的相关知识,文中的示例代码讲解详细,对我们深入学习GO语言有一定的帮助,需要的可以参考下
    2023-11-11
  • 图文详解go语言反射实现原理

    图文详解go语言反射实现原理

    这篇文章主要介绍了图文详解go语言反射实现原理,本文图文并茂给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友参考下吧,需要的朋友可以参考下
    2020-02-02
  • Go语言实现LRU算法的核心思想和实现过程

    Go语言实现LRU算法的核心思想和实现过程

    这篇文章主要介绍了Go语言实现LRU算法的核心思想和实现过程,LRU算法是一种常用的缓存淘汰策略,它的核心思想是如果一个数据在最近一段时间内没有被访问到,那么在将来它被访问的可能性也很小,因此可以将其淘汰,感兴趣想要详细了解可以参考下文
    2023-05-05

最新评论