go语言中goroute使用:=遇到的坑
先看下源代码,预想从1至N总取出所有能被a或b整除的正整数之和,为了利用go语言的并行优势,特使用goroute特性来实现,同时使用普通顺序计算进行效率比较分析
package chango import (
"fmt"
"time"
) func get_sum_of_divisible(num int64, divider int64, resultChan chan int64) {
var sum int64 = 0
var value int64
for value = 1; value < num; value++ {
if value%divider == 0 {
sum += value
}
}
resultChan <- sum
}
func Zhengchu_testing(limit int64, one int64, two int64) {
resultchan := make(chan int64, 3)
var three int64
three = one * two t_start := time.Now() go get_sum_of_divisible(limit, one, resultchan)
go get_sum_of_divisible(limit, two, resultchan)
go get_sum_of_divisible(limit, three, resultchan) one_sum, two_sum, three_sum := <-resultchan, <-resultchan, <-resultchan var sum int64
var value int64
sum = one_sum + two_sum - three_sum fmt.Println(sum)
t_end := time.Now()
fmt.Println(one_sum, two_sum, three_sum)
fmt.Printf("testing times1:%v\n", t_end.Sub(t_start)) sum = 0
t_start = time.Now() for value = 1; value < limit; value++ {
if value%one == 0 {
sum += value }
if value%two == 0 {
sum += value
}
if value%three == 0 {
sum -= value
}
}
fmt.Println(sum)
t_end = time.Now()
fmt.Printf("testing times2:%v\n", t_end.Sub(t_start))
}
然后在main包中调用该chango包
package main import "./chango" func main(){ chango.Zhengchu_testing(10,3,5)
} 初看上去,chango包没有语法毛病,但是实际测试发现,显然通过顺序计算10以内能被3或5整除的所有正整数分别为3,5,6,9,他们之和应该为23,能被3整数的整数之和为3+6+9=18,能被5整除的正整数之和为5,能被3*15=15的正整数之和为0
而通过goroute计算10以内能被3或5整除的所有正整数之和却不是23,而是-13或13,这又是为什么呢?
通过调试发现,问题就出现在“:=”语句中
one_sum, two_sum, three_sum := <-resultchan, <-resultchan, <-resultchan
多次运行
通过分别打印能被3或5或3*5=15的正整数之和,他们的结果竟然是0,18,5或0,5,18,这个结果竟然不唯一 既然是这个语句有问题,此路不通,暂时绕道行之 于是修改源代码如下:
package chango import (
"fmt"
"time"
) func get_sum_of_divisible(num int64, divider int64, resultChan chan int64) {
var sum int64 = 0
var value int64
for value = 1; value < num; value++ {
if value%divider == 0 {
sum += value
}
}
resultChan <- sum
}
func Zhenchu_testing(limit int64, one int64, two int64) {
var three int64
three = one * two
numbers:=[3]int64{one,two,three}
var ch [3](chan int64)
for i := 0; i < 3; i++ {
ch[i] = make(chan int64)
}
t_start := time.Now()
for i:=0;i<3;i++{
go get_sum_of_divisible(limit, numbers[i], ch[i])
}
one_sum:=<-ch[0]
two_sum:=<-ch[1]
three_sum:=<-ch[2]
var sum int64
var value int64
sum = one_sum + two_sum - three_sum
fmt.Println(sum)
t_end := time.Now()
fmt.Println(one_sum, two_sum, three_sum)
fmt.Printf("testing times1:%v\n", t_end.Sub(t_start))
sum = 0
t_start = time.Now()
for value = 1; value < limit; value++ {
if value%one == 0 {
sum += value
}
if value%two == 0 {
sum += value
}
if value%three == 0 {
sum -= value
}
}
fmt.Println(sum)
t_end = time.Now()
fmt.Printf("testing times2:%v\n", t_end.Sub(t_start))
}
这下在main包中调用该chango包通过!
结下来分析goroute同普通顺序计算进行效率比较分析,开始当计算从1至N总取出所有能被a或b整除的正整数之和中N的N的数值比较小的情况下,goroute的并行出来优势没有发挥出来,
测试打印输出如下:
当N=20
78
63 30 15
testing times1:33.557µs
78
testing times2:2.382µs
当N的数值很大的情况下,如100000000,goroute的并行计算优势就发挥出来了!
测试打印输出如下:
2333333316666668
1666666683333333 999999950000000 333333316666665
testing times1:1.574445477s
2333333316666668
testing times2:2.199982414s
参考学习资料
go语言中goroute使用:=遇到的坑的更多相关文章
- Go语言中slice使用注意事项
Go 语言中的slice类型可以理解为是数组array类型的描述符,包含了三个因素: 指向底层数组的指针 slice目前使用到的底层数组的元素个数,即长度 底层数组的最大长度,即容量 因此当我们定义一 ...
- C语言中,头文件和源文件的关系(转)
简单的说其实要理解C文件与头文件(即.h)有什么不同之处,首先需要弄明白编译器的工作过程,一般说来编译器会做以下几个过程: 1.预处理阶段 2.词法与语法分析阶段 3.编译阶段,首先编译成纯汇编语句, ...
- C 语言中 setjmp 和 longjmp
在 C 语言中,我们不能使用 goto 语句来跳转到另一个函数中的某个 label 处:但提供了两个函数——setjmp 和 longjmp来完成这种类型的分支跳转.后面我们会看到这两个函数在处理异常 ...
- c语言中的scanf在java中应该怎么表达,Scanner类。
1 java是面向对象的语言 它没有像C语言中的scanf()函数,但是它的类库中有含有scanf功能的函数 2 java.util包下有Scanner类 Scanner类的功能与scanf类似 3 ...
- C语言中do...while(0)的妙用(转载)
转载来自:C语言中do...while(0)的妙用,感谢分享. 在linux内核代码中,经常看到do...while(0)的宏,do...while(0)有很多作用,下面举出几个: 1.避免goto语 ...
- C语言中,定义的含义?声明的含义?它们之间的区别是什么?
在C语言中,对于定义和声明,也许我们非常的熟悉,但不一定真正的了解! 定义的含义:所谓定义,就是创建(编译器)一个对象,为这个对象分配一块内存空间并取名,也就是我们平常所说的变量名或对象名,一旦这个名 ...
- C++中函数的默认参数和C语言中volatile的学习
1.函数默认参数 1 int func(int a,int b=10) 2 { 3 return a*b; 4 } 5 6 int main() 7 { 8 int c=func(2); 9 cout ...
- C语言中qsort函数用法
C语言中qsort函数用法-示例分析 本文实例汇总介绍了C语言中qsort函数用法,包括针对各种数据类型参数的排序,非常具有实用价值非常具有实用价值. 分享给大家供大家参考.C语言中的qsort ...
- c语言中time相关函数
工作中遇到的函数: int seed = time(NULL); srand(seed); signal(SIGINT, stop); signal(SIGUSR1, sig_usr1); 搜time ...
随机推荐
- Wannafly挑战赛21 E 未来城市规划
传送门 题目中给的信息很难直接维护,但是可以考虑一条边对答案的贡献 在以\(x\)为根的子树里,如果一条边\(i\)的权值为\(w_i\),这条边深度更深的端点为\(to_i\),那么这条边对这个子树 ...
- Python--Virtualenv简明教程
原文链接 http://www.jianshu.com/p/08c657bd34f1 virtualenv通过创建独立Python开发环境的工具, 来解决依赖.版本以及间接权限问题. 比如一个项目依赖 ...
- mysql 案例 ~ pt-archiver 归档工具的使用
一 简介:今天咱们来聊聊pt-archiver的使用 二 相关参数 相关参数1 --statistics 结束的时候给出统计信息:开始的时间点,结束的时间点,查询的行数,归档的行数,删除的行数,以 ...
- linux下mysql 5.7.22 安装
二进制安装 1.下载https://dev.mysql.com/downloads/mysql/5.6.html#downloads 2.官方文档https://dev.mysql.com/doc/r ...
- Three.js基础探寻二——正交投影照相机
本篇主要介绍照相机中的正交投影照相机. 第一篇传送门:Three.js基础探寻一 1.照相机 图形学中的照相机定义了三维空间到二维屏幕的投影方式. 针对投影方式照相机分为正交投影照相机和透视投影照相机 ...
- linux系统切换用户
无权限上传文件解决办法 1.当前登录的普通用户:user1/password1 2.切换到管理员(user2)用户: sudo su - user2 输入user2用户的密码:password2 或者 ...
- Web Services基础学习(W3C)
1.Web services 使用 XML 来编解码数据,并使用 SOAP 来传输数据 2.基础的 Web Services 平台是 XML + HTTP. Web services 平台的元素: S ...
- 【 总结 】Tcp Keepalive 和 HTTP Keepalive 详解
TCP Keepalive Tcp keepalive的起源 双方建立交互的连接,但是并不是一直存在数据交互,有些连接会在数据交互完毕后,主动释放连接,而有些不会,那么在长时间无数据 ...
- ural1989 单点更新+字符串hash
正解是双哈希,不过一次哈希也能解决.. 然后某个数字就对应一个字符串,虽然有些不同串对应同一个数字,但是概率非常小,可以忽略不计.从左到右.从右到左进行两次hash,如果是回文串,那么对应的整数必定存 ...
- MySQL索引底层实现原理
优秀博文: MySQL索引背后的数据结构及算法原理 B树.B-树.B+树.B*树[转],mysql索引 MySQL 和 B 树的那些事 索引的本质 MySQL官方对索引的定义为:索引(Index)是帮 ...