同步协程

通过睡眠方法

// 通过睡眠方式等待

time.Sleep(time.Second)
<-time.NewTimer(time.Second).C
<-time.After(time.Second)
// 睡眠100毫秒

select {
case <- time.NewTimer(100 * time.Millisecond).C:    // 将在100毫秒可读,返回
}

for {
    time.Sleep(10*time.Second)
}

通过WaitGroup同步

// WaitGroup同步: 无需结果

var wg sync.WaitGroup
wg.Add(2)
wg.Done()
wg.Wait() //阻塞等待

通过有缓冲通道同步

// 缓冲为1 的同步

ch := make(chan int, 1)
go func() {
    println(1)
    ch <- 1
}()
<-ch
println(2)
// 缓冲为10,同步
ch := make(chan int, 10)
for i := 0; i < 10; i++ {
    go func() {
        println(1)
        ch <- i
    }()
}
for i := 0; i < cap(ch); i++ {
    <-ch
}
//有缓冲通道: 返回无序结果

func main() {
    ch := make(chan int, 10)
    //开了10个协程让去干活
    for i := 0; i < 10; i++ {
        go func(i int) {
            ch <- i
        }(i)
    }
    //获取10个协程的结果
    res := []int{}
    for i := 0; i < 10; i++ {
        res = append(res, <-ch)
    }
    //打印10个协程的结果(无序)
    fmt.Println(res)
}

参考

// 从100个url获取结果数据, 后集中返回结果(无序)

type User struct {
    name string
    age  int
}

func getUserInfo(resCh chan User, url string) {
    resCh <- User{
        name: url,
        age:  22,
    }
    println(url)
    time.Sleep(time.Second)
}

func main() {
    //获取100个用户的信息

    var users []User //存放结果数据

    urls := []string{
        "http://www.1.com",
        "http://www.2.com",
        "http://www.3.com",
        "http://www.4.com",
        "http://www.5.com",
    }

    resCh := make(chan User, len(urls))
    defer close(resCh)
    for i := 0; i < len(urls); i++ {
        go getUserInfo(resCh, urls[i])
    }

    for i := 0; i < len(urls); i++ {
        users = append(users, <-resCh)
    }

    //打印结果数据
    fmt.Println(users)
}

channel

//判断chan关闭

func main() {
    ch := make(chan int, 10)
    go func() {
        for i := 0; i < 10; i++ {
            ch <- i
        }
        close(ch)
    }()

    //for i := 0; i < 10; i++ {
    //  println(<-ch)
    //}

    //for {
    //  i, ok := <-ch
    //  if !ok {
    //      break
    //  }
    //  println(i)
    //}

    for v := range ch {
        println(v)
    }
}
//有缓冲通道: 控制任务并发数

func main() {
    ch := make(chan int, 3)
    for i := 0; i < 10; i++ {
        go func() {
            ch <- 1
            println("doing")
            time.Sleep(time.Second / 3)
            <-ch
        }()
    }
    time.Sleep(time.Second)
}
// 管道里有数据即输出: 实现优选返回速度快的

func main() {
    ch := make(chan int, 32)
    go func() {
        time.Sleep(time.Second)
        println("google")
        ch <- 1
    }()
    go func() {
        time.Sleep(time.Second/2)
        println("baidu")
        ch <- 1
    }()
    go func() {
        time.Sleep(time.Second/3)
        println("bing")
        ch <- 1
    }()

    println(<-ch)
}

select

//select: 实现任务超时

func main() {
    ch := make(chan int)
    select {
    case <-ch:
    case <-time.After(time.Second):
        println("timeout")
    }
}
//select分支随机执行: 实现生成随机数
func main() {
    ch := make(chan int)
    go func() {
        for{
            select {
            case ch<-0:
            case ch<-1:
            }
            time.Sleep(time.Second/3)
        }
    }()

    for v := range ch {
        println(v)
    }
}
// select的default分支: 实现goroutine的退出
func main() {
    ch := make(chan int)
    go func() {
        for {
            select {
            case <-ch://接收退出信号
            default: //正常执行
                println("hello")
            }
            time.Sleep(time.Second / 3)
        }
    }()
    time.Sleep(time.Second * 3)
    ch <- 1
}
// close chan发广播: 结束多个goroutine
func main() {
    ch := make(chan int)
    for i := 0; i < 10; i++ {
        go func(i int) {
            for {
                select {
                case <-ch: //接收退出信号
                default: //正常执行
                    println("hello", i)
                }
                time.Sleep(time.Second / 3)
            }
        }(i)
    }

    time.Sleep(time.Second * 3)
    close(ch)
}

[go]go并发的更多相关文章

  1. .Net多线程编程—并发集合

    并发集合 1 为什么使用并发集合? 原因主要有以下几点: System.Collections和System.Collections.Generic名称空间中所提供的经典列表.集合和数组都不是线程安全 ...

  2. [ 高并发]Java高并发编程系列第二篇--线程同步

    高并发,听起来高大上的一个词汇,在身处于互联网潮的社会大趋势下,高并发赋予了更多的传奇色彩.首先,我们可以看到很多招聘中,会提到有高并发项目者优先.高并发,意味着,你的前雇主,有很大的业务层面的需求, ...

  3. [高并发]Java高并发编程系列开山篇--线程实现

    Java是最早开始有并发的语言之一,再过去传统多任务的模式下,人们发现很难解决一些更为复杂的问题,这个时候我们就有了并发. 引用 多线程比多任务更加有挑战.多线程是在同一个程序内部并行执行,因此会对相 ...

  4. 关于如何提高Web服务端并发效率的异步编程技术

    最近我研究技术的一个重点是java的多线程开发,在我早期学习java的时候,很多书上把java的多线程开发标榜为简单易用,这个简单易用是以C语言作为参照的,不过我也没有使用过C语言开发过多线程,我只知 ...

  5. 如何在高并发环境下设计出无锁的数据库操作(Java版本)

    一个在线2k的游戏,每秒钟并发都吓死人.传统的hibernate直接插库基本上是不可行的.我就一步步推导出一个无锁的数据库操作. 1. 并发中如何无锁. 一个很简单的思路,把并发转化成为单线程.Jav ...

  6. Java多线程基础——对象及变量并发访问

    在开发多线程程序时,如果每个多线程处理的事情都不一样,每个线程都互不相关,这样开发的过程就非常轻松.但是很多时候,多线程程序是需要同时访问同一个对象,或者变量的.这样,一个对象同时被多个线程访问,会出 ...

  7. 多线程的通信和同步(Java并发编程的艺术--笔记)

    1. 线程间的通信机制 线程之间通信机制有两种: 共享内存.消息传递.   2. Java并发 Java的并发采用的是共享内存模型,Java线程之间的通信总是隐式执行,通信的过程对于程序员来说是完全透 ...

  8. 伪共享(false sharing),并发编程无声的性能杀手

    在并发编程过程中,我们大部分的焦点都放在如何控制共享变量的访问控制上(代码层面),但是很少人会关注系统硬件及 JVM 底层相关的影响因素.前段时间学习了一个牛X的高性能异步处理框架 Disruptor ...

  9. 编写高质量代码:改善Java程序的151个建议(第8章:多线程和并发___建议126~128)

    建议126:适时选择不同的线程池来实现 Java的线程池实现从根本上来说只有两个:ThreadPoolExecutor类和ScheduledThreadPoolExecutor类,这两个类还是父子关系 ...

  10. 理解Storm并发

    作者:Jack47 PS:如果喜欢我写的文章,欢迎关注我的微信公众账号程序员杰克,两边的文章会同步,也可以添加我的RSS订阅源. 注:本文主要内容翻译自understanding-the-parall ...

随机推荐

  1. 转C++之stl::string写时拷贝导致的问题

    前几天在开发某些数据结构到文件的 Dump 和 Load 功能的时候, 遇到的一个 bug . [问题复现] 问题主要出在 Load 过程中,从文件读取数据的时候, 直接使用 fread 的去操作 s ...

  2. 小程序+tgit

    1.微信公众平台-设置-开发者工具  开通 腾讯云和tgit权限管理 如果遇到问题 ..用户二次验证什么的   直接去 腾讯云-安全设置 将“敏感操作”和“异地登陆” 中的保护去掉 2.微信公众平台- ...

  3. BOOST 解析,修改,生成xml样例

    解析XML 解析iworld XML,拿到entity和VisibleVolume的数据 int ParseiWorlds::readXML(const bpath &dir) { ptree ...

  4. [BZOJ 1095] [ZJOI2007]Hide 捉迷藏——线段树+括号序列(强..)

    神做法-%dalao,写的超详细 konjac的博客. 如果觉得上面链接的代码不够优秀好看,欢迎回来看本蒟蒻代码- CODE WITH ANNOTATION 代码中−6-6−6表示左括号'[',用−9 ...

  5. sysbench简易使用

    sysbench简易使用 由于测试需要,需要用到sysbench这个工具.推荐简便使用. # yum 安装 yum install sysbench 创建数据库 CREATE DATABASE `sb ...

  6. HDU 6043 - KazaQ's Socks | 2017 Multi-University Training Contest 1

    /* HDU 6043 - KazaQ's Socks [ 找规律 ] | 2017 Multi-University Training Contest 1 题意: 一共 n 双袜子编号 1~n,每天 ...

  7. postman 跟restsharp 模拟请求http

    https://github.com/restsharp/RestSharp postman 生成的访问代码: 好用! Features Assemblies for .NET 4.5.2 and . ...

  8. logback导入依赖 NoSuchMethodException

    1.我遇到的问题是Spring版本和logback低版本冲突的问题 如何解决:把logback.classic和logback.core等依赖换成1.2.2以上版本的依赖

  9. VUE路由过度效果vs缓存

    app页面 router.js

  10. @RequestMapping的简单理解

    @Controller public class ItemController { @Autowired private ItemService itemService; 获取路径参数.../item ...