文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

一篇文章带你入门Go语言基础之并发

2024-12-03 14:33

关注

前言

Hey,大家好,我是码农星期八,终于到了Go中最牛掰的地方,并发,这也是Go为什么能快速火的原因。

部署方便,不需要容器,随便跑一个都是相当于Nginx的存在,怎么肯能不火

所以,来看看吧!!!

引言

Go语言,专门为并发而生的语言,每启动一个微线程创建一个代价大概2KB起步

假设一个内存条大小4G,一个微线程2kb,1G=1024M=1048576kb,1048576/2=524288,五十多万个

但是你知道像Java,Python等语言,一个线程代价多大吗???,2MB起步,代价直接翻了千倍

所以,激动吧,随便用Go写一个web程序,基本都相当于Nginx

goroutine

Go中的微线程,也叫做goroutine,goroutine是并行处理任务的

就像我用两只手同时操作两个手机打游戏一样

而不是一个手玩玩这个,一个手玩玩那个,这样切换式玩法

goroutine由Go的runtime完成调度,goroutine的本质是在代码(用户态)级别完成的切换,代价很小

像Java,Python等语言的线程,是在操作系统(内核态)级别完成的切花,所以代价非常大

由于goroutine是由runtime完成切换,并且runtime经过Google公司的数位大佬优化,已经很小母牛上山了,牛逼哄哄了。

使用goroutine

在Go中使用goroutine很简单,只需要在想调用的函数前加一个go就行了,这就代表启动了一个goroutine

普通调用函数方式

函数

  1. func Say() { 
  2.     time.Sleep(time.Second
  3.     fmt.Println("我在说话说了1s说完了..."

  main

  1. func main() { 
  2.     //开始时间 
  3.     var start_time = time.Now() 
  4.     //启动10个say说话 
  5.     for i := 0; i < 10; i++ { 
  6.         Say() 
  7.     //结束时间 
  8.     var end_time = time.Now() 
  9.     //计算时间差 
  10.     fmt.Println(end_time.Sub(start_time)) 

执行结果


循环了10次,耗时10s,有点慢啊!

goroutine调用函数方式

函数还是上述的函数

main

  1. func main() { 
  2.     //开始时间 
  3.     var start_time = time.Now() 
  4.     //启动10个say说话 
  5.     for i := 0; i < 10; i++ { 
  6.         go Say() 
  7.     //结束时间 
  8.     var end_time = time.Now() 
  9.     //计算时间差 
  10.     fmt.Println(end_time.Sub(start_time)) 

注意:第6行,前面加了个go关键字,go关键字就表示以一个微线程的方式单独运行这个函数。

执行结果


what??? 0s,什么情况?

为什么会出现0s这种情况

这是因为,在Go中,我们采用的是守护线程的方式,什么意思呢?


在Go中,main函数只要执行完,其他微线程必凉。

就像有的怪兽,他们是互相依赖一个母体的,母体挂了,下面的娃也必挂。

所以该怎么解决这个问题呢???

sync.WaitGroup

上述我们发现,启动了一些微线程,但是微线程还没来得及执行就挂了,是因为main函数跑的太快了,main跑完了,Go运行时自动将其他微线程关闭了。

那反过来想,我们如何让main在最后等一下,等我的孩子们都回来了,我在继续跑。

所以,有一个新的问题,那就是等,祭出法宝sync.WaitGroup

先看一下怎么用

函数

  1. func Say() { 
  2.     //函数结束时取消标记 
  3.     defer wg.Done() 
  4.     //每个函数在启动时加上一个标记 
  5.     wg.Add(1) 
  6.     //函数开始打上一个标记 
  7.     time.Sleep(time.Second*1) 
  8.     fmt.Println("我在说话说了1s说完了..."

main

  1. var wg  sync.WaitGroup 
  2. func main() { 
  3.     //开始时间 
  4.     var start_time = time.Now() 
  5.     //启动10个say说话 
  6.     for i := 0; i < 10; i++ { 
  7.         go Say() 
  8.     // 等待所有标记过的微线程执行完毕 
  9.     wg.Wait() 
  10.     //结束时间 
  11.     var end_time = time.Now() 
  12.     //计算时间差 
  13.     fmt.Println(end_time.Sub(start_time)) 

执行结果


可以看到,10个线程同时启动,1s就完了,并且代码相对简单,就算开启10w个,还是1s多一点

这也是为什么很多公司越来越青睐Go的原因。


runtime.GOMAXPROCS

这个意思要使用多少个核,默认使用全部核心,性能跑满,但是也有意外的情况,

比如一个机器跑了很多其他任务,Go写的这个是不太重要的任务,但是是计算型的,这时候理论来说是不尽量挤兑别人的算力

所以要限制一下当前程序使用电脑的算力

代码

  1. func main() { 
  2.     //本机的cpu个数 
  3.     var cpuNum = runtime.NumCPU() 
  4.     fmt.Println(cpuNum) 
  5.     //设置Go使用cpu个数 
  6.     runtime.GOMAXPROCS(4) 

总结

上述我们学习了Go的并发,学习了

在Go中,轻松实现一个高并发还是挺容易的,但是可能有些不是那么好理解。

 

来源:Go语言进阶学习内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯