当前位置: 首页 > news >正文

【Go】Go语言并发编程:原理、实践与优化

在当今的计算机世界,多核处理器和并发编程已经成为提高程序执行效率的关键。Go语言作为一门极富创新性的编程语言,凭借其强大的并发能力,在这方面表现出色。本文将深入探讨Go语言并发编程的原理,通过实际代码示例展示其应用,并讨论可能的优化策略。

一、并发与并行

在了解Go语言的并发编程之前,我们首先需要理解并发与并行的区别。并发是指程序在逻辑上的同时性,即多个任务在时间上交替执行,但在用户看来,这些任务似乎是在同时进行。并行则是指在物理层面的同时性,即多个任务真正同时执行。

Go语言的并发能力主要体现在其内置的goroutine和channel机制上。Goroutine是一种轻量级的线程,由Go运行时环境管理。通过使用goroutine,我们可以轻松地创建数百万个独立的执行线程,而不会对系统造成过大的开销。

二、Go语言并发编程实现

在Go语言中,我们可以通过以下方式实现并发编程:

使用关键字"go"启动一个新的goroutine:

go func() {// 并行执行的代码
}()

使用channel进行goroutine之间的通信:

ch := make(chan int)  // 创建一个整型通道
go func() {ch <- 42  // 向通道发送数据
}()
fmt.Println(<-ch)  // 从通道接收数据并打印

使用互斥锁(Mutex)保证对共享资源的互斥访问:

var mutex sync.Mutex
var sharedResource int
go func() {mutex.Lock()  // 获取互斥锁sharedResource = 42  // 修改共享资源mutex.Unlock()  // 释放互斥锁
}()
// 在其他地方读取共享资源
fmt.Println(sharedResource)
使用条件变量(Condition Variable)实现goroutine之间的同步:
go
var cond *sync.Cond
var sharedResource int
cond = sync.NewCond(&sync.Mutex{})  // 创建一个条件变量,并关联互斥锁
go func() {sharedResource = 42  // 修改共享资源cond.Signal()  // 发送信号通知等待的goroutine条件已满足
}()
cond.L.Lock()  // 获取互斥锁,并等待条件满足
for sharedResource == 0 {cond.Wait()  // 等待信号通知,继续循环检查条件是否满足
}
fmt.Println(sharedResource)  // 打印共享资源值

三、代码示例:并发爬虫程序

下面是一个使用Go语言实现并发爬虫的简单示例。该程序通过启动多个goroutine并发地获取网页内容,并将结果存储在一个channel中。最后,主goroutine从channel中读取数据并处理。

package mainimport ("fmt""net/http""sync"
)func fetch(url string, ch chan<- string, wg *sync.WaitGroup) {defer wg.Done()resp, err := http.Get(url)if err != nil {return}defer resp.Body.Close()ch <- resp.Text()  // 将网页内容发送到channel中
}func main() {urls := []string{"http://example.com", "http://example.org", "http://example.net"}ch := make(chan string)  // 创建一个字符串通道用于接收网页内容var wg sync.WaitGroup    // 用于等待所有爬虫任务完成的计数器wg.Add(len(urls))         // 设置计数器的初始值for _, url := range urls {go fetch(url, ch, &wg)  // 启动每个爬虫任务,并将结果发送到channel中}go func() {   // 创建一个辅助goroutine,用于等待所有爬虫任务完成并关闭channelwg.Wait()close(ch)     // 关闭channel,表示所有数据已发送完毕}()for data := range ch {  // 从channel中读取每个网页的内容并处理fmt.Println(data)   // 这里只是简单地打印网页内容,实际应用中可以根据需求进行进一步处理和存储等操作。}
}
http://www.lryc.cn/news/140489.html

相关文章:

  • HTTPS协议加密原理
  • L1-034 点赞(Python实现) 测试点全过
  • MySQL 存储过程 循环处理数据 while repeat
  • 基于配置类方式管理 Bean
  • 最新CMS指纹识别技术
  • 快速入门学习记录:常用代码、特定函数、复杂概念和特定功能说明
  • 【win视频播放器】HEVC视频扩展
  • React+Typescript 父子组件事件传值
  • python人工智能和机器学习
  • [PyTorch][chapter 51][Auto-Encoder -1]
  • Uniapp或者微信小程序如何动态的计算Scrollview的高度
  • Abase数据库管理系统
  • 系统架构设计高级技能 · 大数据架构设计理论与实践
  • ubuntu上使用osg3.2+osgearth2.9
  • C语言巧妙打印64位整数
  • c语言每日一练(11)
  • SLAM十四讲学习笔记 第二期:部分课后实践代码
  • Android kotlin 跳转手机热点开关页面和判断热点是否打开
  • Redis 执行 RDB 快照期间,主进程可以正常处理命令吗?
  • Python加入Excel--生产力大提高|微软的全方面办公
  • Excel 分组排名
  • Redis初始以及安装
  • react导出、导入文件
  • (一)Redis——String
  • Spring Boot多环境指定yml或者properties
  • MinDoc:针对IT团队的文档、笔记系统
  • 【分享】华为设备登录安全配置案例
  • Starrocks与MySQL函数的区别
  • 【HTML】学习笔记(自用持续更新)
  • 代码随想录训练营第四十二天|0-1背包理论基础(一)、0-1背包理论基础(二)、416分割等和子集