下载APP
登录
关闭
讲堂
算法训练营
Python 进阶训练营
企业服务
极客商城
客户端下载
兑换中心
渠道合作
推荐作者
当前播放: 33 | 对象池
00:00 / 00:00
标清
  • 标清
1.0x
  • 2.0x
  • 1.5x
  • 1.25x
  • 1.0x
  • 0.5x
网页全屏
全屏
00:00
付费课程,可试看

Go语言从入门到实战

共55讲 · 55课时,约700分钟
6137
免费
01 | Go语言课程介绍
免费
02 | 内容综述
免费
03 | Go语言简介:历史背景、发...
免费
04 | 编写第一个Go程序
免费
05 | 变量、常量以及与其他语言...
06 | 数据类型
07 | 运算符
08 | 条件和循环
09 | 数组和切片
10 | Map声明、元素访问及遍历
11 | Map与工厂模式,在Go语言...
12 | 字符串
13 | Go语言的函数
14 | 可变参数和defer
15 | 行为的定义和实现
16 | Go语言的相关接口
17 | 扩展与复用
18 | 不一样的接口类型,一样的...
19 | 编写好的错误处理
20 | panic和recover
21 | 构建可复用的模块(包)
22 | 依赖管理
23 | 协程机制
24 | 共享内存并发机制
25 | CSP并发机制
26 | 多路选择和超时
27 | channel的关闭和广播
28 | 任务的取消
29 | Context与任务取消
30 | 只运行一次
31 | 仅需任意任务完成
32 | 所有任务完成
33 | 对象池
34 | sync.pool对象缓存
35 | 单元测试
36 | Benchmark
37 | BDD
38 | 反射编程
39 | 万能程序
40 | 不安全编程
41 | 实现pipe-filter framew...
42 | 实现micro-kernel frame...
43 | 内置JSON解析
44 | easyjson
45 | HTTP服务
46 | 构建RESTful服务
47 | 性能分析工具
48 | 性能调优示例
49 | 别让性能被锁住
50 | GC友好的代码
51 | 高效字符串连接
52 | 面向错误的设计
53 | 面向恢复的设计
54 | Chaos Engineering
55 | 结束语
本节摘要
展开

精选留言(9)

  • 2019-03-27
    一楼[ZY]的留言 “当一个channel满了之后 无法继续插入 会报错,但是放到协程里面却可以在满了之后 继续插入数据 为什么呢?”
    我的理解是这样的:
    1.你的代码里有生产者和消费者,使用for循环开启了10个协程,channel还没满的时候就被消费者提前消费掉了,其实只能添加10数据,可以把消费者去掉, 并且加上日志打印:代码如下:
        var wg sync.WaitGroup
        num := 10
        ch := make(chan int, num)
        for i := 0; i < 1000; i++ {
            wg.Add(1)
            go func(i int) {
                ch <- i    // 协程满后会阻塞,导致无法执行厦门的wg.Done()导致死锁报异常
                fmt.Printf("执行协程:%d\n",i)
                wg.Done()
            }(i)
            wg.Wait()
        }

    2.可以在一个协程里循环添加超过channel大小的数据,依旧会阻塞
    func TestChannelPush(t *testing.T) {
        ch := make(chan int, 10)
        var wg sync.WaitGroup
        wg.Add(1)
        pushChannel(ch,&wg)
        wg.Wait()
    }

    func pushChannel(ch chan int, wg *sync.WaitGroup) {
        go func() {
            for i := 0; i < 100; i++ {
                ch <- i
            }
            wg.Done()
        }()
    }

    因为这里用了WaitGroup,所以会有死锁,实际上是会阻塞。
    展开
    4
  • 2019-03-27
    蔡老师您好
    这节课您里面讲到 当一个channel满了之后 无法继续插入 会报错
    但是放到协程里面却可以在满了之后 继续插入数据 为什么呢?
    申请的时候是10个 插入11个

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

    total := 0
    for i := 0; i < num+1; i++ {
        total += <-ch
    }
    fmt.Println("for total-------", total)
    展开
    3
  • 2019-03-28
    感觉还是带着写代码比较容易理解
    2
  • 2019-03-27
    补充一下ZY的留言,channel大小虽然只有10,for循环开启11个协程往channel添加数据,但是因为协程和主程序是异步的所以看不出阻塞,因为后面的消费者程序把channel的数据消费了,所以上面可能存在的阻塞又通了,一边生产一边消费,MQ经常堆包也是类似的情况,在主机内存空间有限的情况下如果消费者处理消息的速度比生产者慢,就有可能堆包,严重的就阻塞。
    2
  • 2019-04-04
    蔡老师您好
    在对象池一课我参考之前的课程使用unsafe.Pointer在不放回的情况下打出了每个对象的地址,发现他们都是一样的, 这个时候结构体里没有定义任何成员。
    我把NewObjPool的初始化对象改成了每个都赋予一个id int子成员的情况下10个对象打印出的地址就不同了(预期)
    后来在网上查阅发现这个是空结构体,用unsafe.Sizeof发现占用的大小为0,所以每次初始化的位移就是0,所以每个对象的地址都是相同。
    而如果增加一个int成员的话,Size就会变成8,然后地址的位移也是8.

    另外老师github上的代码没有更新了最近。。。
    展开

    作者回复: 很赞!在可执行的代码中学习!
    代码我会尽快提交,并通知平台更新。

    1
  • 2019-07-18
    老师,假如我用对象池当做MySQL连接池用的话,我创建100个连接,然后在并发情况下连接池空了,这个时候后续的query只能等待或者timeout吗?还有没有别的更好的方案

    作者回复: 较常见的,还可以降级处理,超时后直接从缓存返回历史数据或返回默认数据。

  • 2019-04-01
    老师你好,time out在项目中的正确处理是什么?应该不是只报个错,然后输出timeout吧。。

    作者回复: 后面的高可用性课程会讲。继续看哦

  • 2019-03-28
    这就结束啦,等等等更新
  • 2019-03-28
    感谢 蓝士钦 的回答
    确实是这样的 带长度的channel 在满了之后 会阻塞
    没满的时候不会阻塞