利用多核 CPU 实现并行计算

由 学院君 创建于1年前, 最后更新于 3个月前

版本号 #2

5814 views

2 likes

1 collects

开始之前,我们先澄清两个概念,「多核」指的是有效利用 CPU 的多核提高程序执行效率,「并行」和「并发」一字之差,但其实是两个完全不同的概念,「并发」一般是由 CPU 内核通过时间片或者中断来控制的,遇到 IO 阻塞或者时间片用完时会交出线程的使用权,从而实现在一个内核上处理多个任务,而「并行」则是多个处理器或者多核处理器同时执行多个任务,同一时间有多个任务在调度,因此,一个内核是无法实现并行的,因为同一时间只有一个任务在调度:

多进程、多线程以及协程显然都是属于「并发」范畴的,可以实现程序的并发执行,至于是否支持「并行」,则要看程序运行系统是否是多核,以及编写程序的语言是否可以利用 CPU 的多核特性。

下面我们以 goroutine 为例,来演示如何在 Go 语言中通过协程有效利用「多核」实现程序的「并行」执行,具体实现的话就是根据系统 CPU 核心数量来分配等值的子协程数,让所有协程分配到每个内核去并行执行。要查看系统核心数,以 MacOS 为例, 可以通过 sysctl hw 命令分别查看物理 CPU 和逻辑 CPU 核心数:

我的系统物理 CPU 核心数是 4 个,逻辑 CPU 核心数是 8 个,所谓物理 CPU 核心数指的是真正插在物理插槽上 CPU 的核心数,逻辑 CPU 核心数指的是结合 CPU 多核以及超线程技术得到的 CPU 核心数,最终核心数以逻辑 CPU 核心数为准。

此外,你也可以在 Go 语言中通过调用 runtime.NumCPU() 方法获取 CPU 核心数。

接下来,我们来模拟一个可以并行的计算任务:启动多个子协程,子协程数量和 CPU 核心数保持一致,以便充分利用多核并行运算,每个子协程计算分给它的那部分计算任务,最后将不同子协程的计算结果再做一次累加,这样就可以得到所有数据的计算总和。我们编写对应的示例文件 parallel.go:

package main

import (

"fmt"

"runtime"

"time"

)

func sum(seq int, ch chan int) {

defer close(ch)

sum := 0

for i := 1; i <= 10000000; i++ {

sum += i

}

fmt.Printf("子协程%d运算结果:%d\n", seq, sum)

ch

}

func main() {

// 启动时间

start := time.Now()

// 最大 CPU 核心数

cpus := runtime.NumCPU()

runtime.GOMAXPROCS(cpus)

chs := make([]chan int, cpus)

for i := 0; i < len(chs); i++ {

chs[i] = make(chan int, 1)

go sum(i, chs[i])

}

sum := 0

for _, ch := range chs {

res :=

sum += res

}

// 结束时间

end := time.Now()

// 打印耗时

fmt.Printf("最终运算结果: %d, 执行耗时(s): %f\n", sum, end.Sub(start).Seconds())

}

这里我们通过 runtime.NumCPU() 获取逻辑 CPU 核心数,然后通过 runtime.GOMAXPROCS() 方法设置程序运行时可以使用的最大核心数,这里设置为和系统 CPU 核心数一致,然后初始化一个通道数组,数量和 CPU 核心数保持一致,以便充分利用多核实现并行计算,接下来就是依次启动子协程进行计算,并在子协程中计算完成后将结果数据发送到通道中,最后在主协程中接收这些通道数据并进行再次累加,作为最终计算结果打印出来,同时计算程序运行时间作为性能的考量依据。

此时,我们运行 parallel.go,得到的结果如下:

然后我们修改 runtime.GOMAXPROCS() 方法中传入的 CPU 核心数为 1,再次运行 parallel.go,得到的结果如下:

可以看到使用多核比单核整体运行速度快了4倍左右,查看系统 CPU 监控也能看到所有内核都被打满,这在 CPU 密集型计算中带来的性能提升还是非常显著的,不过对于 IO 密集型计算可能没有这么显著,甚至有可能比单核低,因为 CPU 核心之间的切换也是需要时间成本的,所以 IO 密集型计算并不推荐使用这种机制,什么是 IO 密集型计算?比如数据库连接、网络请求等。

另外,需要注意的是,目前 Go 语言默认就是支持多核的,所以如果上述示例代码中没有显式设置 runtime.GOMAXPROCS(cpus) 这行代码,编译器也会利用多核 CPU 来执行代码,其结果是运行耗时和设置多核是一样的。

mysql如何使用多核cpu_利用多核 CPU 实现并行计算相关推荐

  1. 如何利用计算机多核,如何利用多核电脑实现Matlab的并行运算

    2017-05-28 回答 matlab并行运算 目前,新购置的电脑大部分都是多核的了,使用matlab进行大量计算时如何有效利用多核呢?matlab目前版本已经比较好的支持多核并行运算了.是用的ma ...

  2. linux多核操作命令,利用多核CPU加速你的Linux命令

    你是否曾经有过要计算一个非常大的数据(几百GB)的需求?或在里面搜索,或其它操作--一些无法并行的操作.数据专家们,我是在对你们说.你可能有一个4核或更多核的CPU,但我们合适的工具,例如 grep, ...

  3. python多线程多个cpu_为什么python的多线程不能利用多核CPU?

    为什么python的多线程不能利用多核CPU,但是咱们在写代码的时候,多线程的确是在并发,而且还比单线程快. 一.python的多线程不能利用多核CPU? 原因: 因为GIL,python只有一个GI ...

  4. 如何利用多核CPU提高虚拟现实性能?

    虚拟现实引领新时代令人惊喜的新体验,但也带来了各种新的挑战.其中之一就是虚拟现实应用"耗能"问题. 虚拟现实挑战了图形和仿真技术,以至于创造优秀虚拟现实体验的硬件需求成为日前热点. ...

  5. python多线程不能利用多核cpu,但有时候多线程确实比单线程快。

    python 为什么不能利用多核 CPU  GIL 其实是因为在 python中有一个 GIL( Global Interpreter Lock),中文为:全局解释器锁.  1.最开始时候设计GIL是 ...

  6. 如何利用多核CPU来加速你的Linux命令

    如何利用多核CPU来加速你的Linux命令 原文出处: rankfocus   译文出处: 外刊IT评论 你是否曾经有过要计算一个非常大的数据(几百GB)的需求?或在里面搜索,或其它操作--一些无法并 ...

  7. python如何指定使用的cpu核_Python如何利用多核处理器

    python多线程不能利用多核cpu,但有时候多线程确实比单线程快. python 为什么不能利用多核 CPU  GIL 其实是因为在 python中有一个 GIL( Global Interpret ...

  8. python3 多线程_图解|为什么 Python 多线程无法利用多核

    (给Python开发者加星标,提升Python技能) 来源:后端技术指南针 1.全局解释锁 如题: Python的多线程为什么不能利用多核处理器? 全局解释器锁(Global Interpreter ...

  9. ITK:对多个线程上的数据进行操作以利用多核处理器

    ITK:对多个线程上的数据进行操作以利用多核处理器 ITK:对多个线程上的数据进行操作以利用多核处理器 内容提要 结果 C++实现代码 ITK:对多个线程上的数据进行操作以利用多核处理器 内容提要 内 ...

最新文章

  1. 《Android App开发入门:使用Android Studio 2.X开发环境》——1-3 Android Studio 快速上手...
  2. ryu的防火墙功能 ryu.app.rest_firewall,配合mininet和open vswitch(OVS)
  3. 运维工程师必备之MySQL数据的主从复制、半同步复制和主主复制详解
  4. 对MVC、MVP、MVVM的理解
  5. 华为手机asph啥机型_华为正式宣布!19款机型开启新系统内测,你的手机榜首有名吗?...
  6. boost::graph模块实现dijkstra算法的测试程序
  7. 阿里巴巴JAVA开发手册及开发插件
  8. IIS负载均衡-Application Request Route详解第三篇:使用ARR进行Http请求的负载均衡
  9. Java 学习总结(189)—— Java 8 功能接口使用总结
  10. fgets,cin. getline被跳过
  11. Java中的类型列表与类型ArrayList
  12. 博客园电子期刊2010年2月刊发布啦
  13. xp计算机u盘重装系统,用U盘给旧电脑重装XP系统,自己摸索的方法:两大步骤、两个关键...
  14. HTML插入Flash的全兼容完美解决方案-SWFObject
  15. 9个你可能最想了解的关于微信指数的问题
  16. 路由器信号总是无法与手机连接服务器,手机搜不到路由器信号_手机无法搜到路由器信号怎么办?-192路由网...
  17. 数据分析必看的oCPM/oCPC核心知识点
  18. iOS 10诸如相机、相册、通讯录、麦克风、定位权限设置,防止奔溃或上架被拒
  19. 启动virtualbox虚拟机显示Attempted to kill the idle task错误
  20. 如何处理高并发和单点故障

热门文章

  1. PHP孟加拉钢厂_昆钢推进孟加拉国、柬埔寨、缅甸钢铁国际产能合作示范园区建设...
  2. 2021-2025年中国拜勒病治疗行业市场供需与战略研究报告
  3. 诸葛新增快应用SDK,满足客户更多数据采集方案支持!
  4. MySQL必知必会(1)
  5. 将doc文档生成html页面
  6. Tomcat:Tomcat网站上的core和deployer的区别
  7. 【C生万物】 函数递归篇
  8. SQL零基础入门学习(七)
  9. http报错405问题解决方法
  10. 基于word2vec+TextCNN 实现中文文本分类