Go并发编程(四)
多进程
多线程基于回调的非阻塞/异步IO协程
协程
goroutine
package mainimport"fmt"func Add(x, y int){z := x + yfmt.Println(z)}func main(){for i :=0; i <10; i++{go Add(i, i)}}
var count intfunc Count(lock*sync.Mutex){lock.Lock()count++fmt.Println(count)lock.Unlock()}func main(){lock:=&sync.Mutex{}for i :=0; i <10; i++{go Count(lock)}for{lock.Lock()c := countlock.Unlock()runtime.Gosched()if c >10{break}}}
channel
channel是Go语言在语言级别提供的goroutine间的通信方式。
//声明一个chanvarch chan intvar mch map[string]chan bool//声明并初始化一个int类型的chanchan1 := make(chan int,1)//将一个数据写入channel中chan1 <-1getchan1 :=<-chan1
chan1 := make(chan int,1)//将1写入channel中chan1 <-1//将一个数据从channel中读取到getchan1中getchan1 :=<-chan1fmt.Println(getchan1) //输出1
select
通过调用select()函数来监控一系列的文件句柄。一旦其中一个文件句柄发生了IO动作,该select()调用就会被返回
select{case<-chan1:// 如果chan1成功读到数据,则进行该case处理语句case chan2 <-1:// 如果成功向chan2写入数据,则进行该case处理语句default:// 如果上面都没有成功,则进入default处理流程}
ch := make(chan int,1)for{select{case ch <-0:case ch <-1:}i :=<-chfmt.Println("Value received:", i)}
//创建一个带缓冲的channelc := make(chan int,1024)
// 首先,我们实现并执行一个匿名的超时等待函数timeout := make(chan bool,1)go func(){time.Sleep(1e9)// 等待1秒钟timeout <-true}()// 然后我们把timeout这个channel利用起来select{case<-ch:// 从ch中读取到数据case<-timeout:// 一直没有从ch中读取到数据,但从timeout中读取到了数据}
channel的传递
type PipeDatastruct{value inthandler func(int)intnext chan int}
func handle(queue chan *PipeData){for data := range queue {data.next<- data.handler(data.value)}}
var ch1 chan int// ch1是一个正常的channel,不是单向的var ch2 chan<- float64// ch2是单向channel,只用于写float64数据var ch3 <-chan int// ch3是单向channel,只用于读取int数据
//单项channel初始化,ch4被转换为一个单项读channel和一个单向写channelch4 := make(chan int)ch5 :=<-chan int(ch4)// ch5就是一个单向的读取channelch6 := chan<-int(ch4)// ch6 是一个单向的写入channel
close(ch)
x, ok :=<-ch //返回值是false则表示ch已经被关闭。
多核并行化
type Vector[]float64// 分配给每个CPU的计算任务func (v Vector)DoSome(i, n int, u Vector, c chan int){for; i < n; i++{v[i]+= u.Op(v[i])}c <-1// 发信号告诉任务管理者我已经计算完成了}const NCPU =16// 假设总共有16核func (v Vector)DoAll(u Vector){c := make(chan int, NCPU)// 用于接收每个CPU的任务完成信号for i :=0; i < NCPU; i++{go v.DoSome(i*len(v)/NCPU,(i+1)*len(v)/NCPU, u, c)}// 等待所有CPU的任务完成for i :=0; i < NCPU; i++{<-c // 获取到一个数据,表示一个CPU计算完成了}// 到这里表示所有计算已经结束}
GOMAXPROCS只是个时间问题。
runtime.Gosched()
典型使用模式如下:
var l sync.Mutexfunc foo(){l.Lock()defer l.Unlock()//...}
全局唯一性操作
var a stringvar once sync.Oncefunc setup(){a ="hello, world"}func doprint(){once.Do(setup)print(a)}func twoprint(){go doprint()go doprint()}
once.Do()调用结束后才继续。
Go并发编程(四)的更多相关文章
- 【Java并发编程四】关卡
一.什么是关卡? 关卡类似于闭锁,它们都能阻塞一组线程,直到某些事件发生. 关卡和闭锁关键的不同在于,所有线程必须同时到达关卡点,才能继续处理.闭锁等待的是事件,关卡等待的是其他线程. 二.Cycli ...
- Java 并发编程(四):如何保证对象的线程安全性
01.前言 先让我吐一句肺腑之言吧,不说出来会憋出内伤的.<Java 并发编程实战>这本书太特么枯燥了,尽管它被奉为并发编程当中的经典之作,但我还是忍不住.因为第四章"对象的组合 ...
- 并发编程>>四种实现方式(三)
概述 1.继承Thread 2.实现Runable接口 3.实现Callable接口通过FutureTask包装器来创建Thread线程 4.通过Executor框架实现多线程的结构化,即线程池实现. ...
- Java并发编程 (四) 线程安全性
个人博客网:https://wushaopei.github.io/ (你想要这里多有) 一.线程安全性-原子性-atomic-1 1.线程安全性 定义: 当某个线程访问某个类时,不管运行时环境 ...
- Java并发编程(四):并发容器(转)
解决并发情况下的容器线程安全问题的.给多线程环境准备一个线程安全的容器对象. 线程安全的容器对象: Vector, Hashtable.线程安全容器对象,都是使用 synchronized 方法实现的 ...
- 并发编程(四):ThreadLocal从源码分析总结到内存泄漏
一.目录 1.ThreadLocal是什么?有什么用? 2.ThreadLocal源码简要总结? 3.ThreadLocal为什么会导致内存泄漏? 二.ThreadLoc ...
- java并发编程的艺术——第四章总结
第四章并发编程基础 4.1线程简介 4.2启动与终止线程 4.3线程间通信 4.4线程应用实例 java语言是内置对多线程支持的. 为什么使用多线程: 首先线程是操作系统最小的调度单元,多核心.多个线 ...
- 并发编程(四):atomic
本篇博客我们主要讲述J.U.C包下的atomic包,在上篇博客"并发模拟"的最后,我们模拟高并发的情形时出现了线程安全问题,怎么解决呢?其实解决的办法有很多中,如直接在add()方 ...
- Python并发编程之线程消息通信机制任务协调(四)
大家好,并发编程 进入第四篇. 本文目录 前言 Event事件 Condition Queue队列 总结 . 前言 前面我已经向大家介绍了,如何使用创建线程,启动线程.相信大家都会有这样一个想法,线程 ...
随机推荐
- 编译和运行java文件 找不到或无法加载主类
这边提供一个关于程序中含有package关键字,使用“终端”运行程序时出现“找不到或无法加载主类”,而使用Eclipse软件可以正常运行程序的可能解决办法. 例如程序名为HelloWorldTest. ...
- 明明想在学校中请一些同学一起做一项问卷调查,为了实验的客观性,他先用计算机生成了N个1到1000之间的随机整数(N≤1000),对于其中重复的数字,只保留一个,把其余相同的数去掉,不同的数对应着不同的学生的学号。然后再把这些数从
明明想在学校中请一些同学一起做一项问卷调查,为了实验的客观性,他先用计算机生成了N个1到1000之间的随机整数(N≤1000),对于其中重复的数字,只保留一个,把其余相同的数去掉,不同的数对应着不同的 ...
- qualcomm batch 烧录脚本
在烧录android系统候用到了windows的批处理文件,拿出来分析一下,顺便记录一下高通平台烧录系统的命令. @echo off :: @ :不显示后面的命令,就是后面的"echo of ...
- SQL数据查询之——单表查询
一.SQL数据查询的一般格式 数据查询是数据库的核心操作.SQL提供了SELECT语句进行数据查询,其一般格式为: SELECT [ALL | DISTINCT]<目标列表达式>[,< ...
- 第三百二十二节,web爬虫,requests请求
第三百二十二节,web爬虫,requests请求 requests请求,就是用yhthon的requests模块模拟浏览器请求,返回html源码 模拟浏览器请求有两种,一种是不需要用户登录或者验证的请 ...
- 关于eclipse导工程或移植工程常碰到的错误汇总
在开发过程中,eclipse是使用得最多的IDE,但由于其开源且免费的性质决定了其不然有很多的BUG,在项目很赶的时期碰到某些很恶的错误很浪费时间,也很让人郁闷,现我总结一下我碰到的错误并总结下对 ...
- iOS项目的目录结构(Cocoa China)
目录结构 AppDelegate Models Macro General Helpers Vendors Sections Resources 一个合理的目录结构首先应该是清晰的,让人一眼看上去 ...
- e768. 创建单选按钮
// Create an action for each radio button Action action1 = new AbstractAction("RadioButton Labe ...
- (转) 解密H264、AAC硬件解码的关键扩展数据处理
出自:http://blog.itpub.net/30168498/viewspace-1576794/ 通过上一篇文章,我们用ffmpeg分离出一个多媒体容器中的音视频数据,但是很可能这 ...
- php可选缓存APC
1.APC缓存简介 APC,全称是Alternative PHP Cache,官方翻译叫”可选PHP缓存”.它为我们提供了缓存和优化PHP的中间代码的框架. APC的缓存分两部分:系统缓存和用户数据缓 ...