Go语言并发模式

利用goroutine和channel进行go的并发模式,实现一个资源池实例(《Go语言实战》书中实例稍作修改)

资源池可以存储一定数量的资源,用户程序从资源池获取资源进行使用,使用完成将资源释放回资源池

程序

pool.go

package pool

import (
"errors"
"io"
"log"
"sync"
"time"
) type Pool struct {
m sync.Mutex
resource chan io.Closer
//创建资源的方法,由用户程序自己生成传入
factory func() (io.Closer, error)
closed bool
//资源池获取资源超时时间
timeout <-chan time.Time
} //资源池关闭标志
var ErrPoolClosed = errors.New("资源池已经关闭")
//超时标志
var ErrTimeout = errors.New("获取资源超时") //新建资源池
func New(fn func() (io.Closer, error), size int) (*Pool, error) {
if size <= 0 {
return nil, errors.New("新建资源池大小太小")
}
//新建资源池
p := Pool{
factory: fn,
resource: make(chan io.Closer, size),
}
//向资源池循环添加资源,直到池满
for count := 1; count <= cap(p.resource); count++ {
r, err := fn()
if err != nil {
log.Println("添加资源失败,创建资源方法返回nil")
break
}
log.Println("资源加入资源池")
p.resource <- r
}
log.Println("资源池已满,返回资源池")
return &p, nil
} //获取资源
func (p *Pool) Acquire(d time.Duration) (io.Closer, error) {
//设置d时间后超时
p.timeout = time.After(d)
select {
case r, ok := <-p.resource:
log.Println("获取", "共享资源")
if !ok {
return nil, ErrPoolClosed
}
return r, nil
case <-p.timeout:
return nil, ErrTimeout
}
} //放回资源池
func (p *Pool) Release(r io.Closer) {
//上互斥锁,和Close方法对应,不同时操作
p.m.Lock()
defer p.m.Unlock() if p.closed {
r.Close()
return
}
//资源放回队列
select {
case p.resource <- r:
log.Println("资源放回队列")
default:
log.Println("资源队列已满,释放资源")
r.Close()
}
} //关闭资源池
func (p *Pool) Close() {
//互斥锁,保证同步,和Release方法相关,用同一把锁
p.m.Lock()
defer p.m.Unlock() if p.closed {
return
}
p.closed = true
//清空通道资源之前,将通道关闭,否则引起死锁
close(p.resource)
for r := range p.resource {
r.Close()
}
}

main.go

package main

import (
"gopro/patterns/pool"
"io"
"log"
"math/rand"
"sync"
"sync/atomic"
"time"
) const (
maxGoroutines = 25
pooledResources = 2
) //实现接口类型 资源类型
type dbConnection struct {
ID int32
} //实现接口方法
func (conn *dbConnection) Close() error {
log.Printf("资源关闭,ID:%d\n", conn.ID)
return nil
} //给每个连接资源给id
var idCounter int32
//创建新资源
func createConnection() (io.Closer, error) {
id := atomic.AddInt32(&idCounter, 1)
log.Printf("创建新资源,id:%d\n", id)
return &dbConnection{ID: id}, nil
} //测试资源池
func performQueries(query int, p *pool.Pool) {
conn, err := p.Acquire(10 * time.Second)
if err != nil {
log.Println("获取资源超时")
log.Println(err)
return
}
//方法结束后将资源放进资源池
defer p.Release(conn)
//模拟使用资源
time.Sleep(time.Duration(rand.Intn(1000)) * time.Millisecond)
log.Printf("查询goroutine id:%d,资源ID:%d\n", query, conn.(*dbConnection).ID)
} func main() {
var wg sync.WaitGroup
wg.Add(maxGoroutines) p, err := pool.New(createConnection, pooledResources)
if err != nil {
log.Println(err)
} //每个goroutine一个查询,每个查询从资源池中获取资源
for query := 0; query < maxGoroutines; query++ {
go func(q int) {
performQueries(q, p)
wg.Done()
}(query)
} //主线程等待
wg.Wait()
log.Println("程序结束")
//释放资源
p.Close()
}

执行结果

循环使用两个资源

// :: 创建新资源,id:
// :: 资源加入资源池
// :: 创建新资源,id:
// :: 资源加入资源池
// :: 资源池已满,返回资源池
// :: 获取 共享资源
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 程序结束
// :: 资源关闭,ID:
// :: 资源关闭,ID:

超时结果

修改超时时间为很短

conn, err := p.Acquire( * time.Second)

结果:

// :: 创建新资源,id:
// :: 资源加入资源池
// :: 创建新资源,id:
// :: 资源加入资源池
// :: 资源池已满,返回资源池
// :: 获取 共享资源
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 获取 共享资源
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 获取资源超时
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 查询goroutine id:,资源ID:
// :: 资源放回队列
// :: 程序结束
// :: 资源关闭,ID:
// :: 资源关闭,ID:

Go语言-并发模式-资源池实例(pool)的更多相关文章

  1. Go语言-并发模式-goroutine池实例(work)

    介绍 使用无缓冲的通道来创建一个 goroutine 池,这些 goroutine 执行并控制一组工作,让其并发执行.在这种情况下,使用无缓冲的通道要比随意指定一个缓冲区大小的有缓冲的通道好,因为这个 ...

  2. 《Go语言实战》摘录:7.2 并发模式 - pool

    7.2 并发模式 - pool

  3. WCF实例上下文模式与并发模式对性能的影响

    实例上下文模式 InstanceContextMode 控制在响应客户端调用时,如何分配服务实例.InstanceContextMode 可以设置为以下值: •Single – 为所有客户端调用分配一 ...

  4. go语言】Goroutines 并发模式

    并发模式 让我们先来回顾一下boring函数的例子. func boring(msg string, c chan string) {    for i := 0; ; i++ {         c ...

  5. 《Go语言实战》摘录:7.3 并发模式 - work

    7.3 并发模式 - work

  6. 《Go语言实战》摘录:7.1 并发模式 - runner

    7.1 并发模式 - runner

  7. 《C#并发编程经典实例》笔记

    1.前言 2.开宗明义 3.开发原则和要点 (1)并发编程概述 (2)异步编程基础 (3)并行开发的基础 (4)测试技巧 (5)集合 (6)函数式OOP (7)同步 1.前言 最近趁着项目的一段平稳期 ...

  8. Go并发模式:管道与取消

    关键字:Go语言,管道,取消机制,并发,sync.WaitGroup,包引用,通道,defer,select GO并发模式:管道与取消 简介 Go的并发能力可以使构建一个流数据管道变得非常容易,并且可 ...

  9. 09. Go 语言并发

    Go 语言并发 并发指在同一时间内可以执行多个任务.并发编程含义比较广泛,包含多线程编程.多进程编程及分布式程序等.本章讲解的并发含义属于多线程编程. Go 语言通过编译器运行时(runtime),从 ...

随机推荐

  1. C++远征--jame_yuan(慕课网)

    int &a=b;引用符 int const a = 3; int *p = &a;     只读 不能 赋给 读写 有默认参数值得参数必须在参数表的最右端 void fun(int ...

  2. DCGAN生成目标训练图片

    前言: GAN的原理很简单,但是它有很多变体,如:DCGAN.CycleGAN.DeblurGAN等,它们也被用在不同地方,本文将用到DCGAN来生成头像图片,可以做到以假乱真的地步. 1.首先调用程 ...

  3. jsp与spring mvc后台controller间参数传递处理之总结

    在编程过程中,最容易出现问题及卡壳的地方,往往是各层之间接缝处,接缝处往往存在着各种各样的参数传递,数据转换和格式化,参数很好的传递并正确接收过来之后就是复杂逻辑之间的处理了,所以为了避免多种问题占用 ...

  4. Python 列表/元组/字典总结

    序列是Python中最基本的数据结构.序列中的每个元素都分配一个数字 - 它的位置,或索引,第一个索引是0,第二个索引是1,依此类推. Python有6个序列的内置类型,但最常见的是列表和元组. 序列 ...

  5. Node.js NPM 教程

    NPM是Node.js的包(或模块)管理器,是Node.js应用程序开发的核心. www.npmjs.com上有海量的Node.js包,供免费下载使用. 当安装Node.js时,NPM程序会被同时安装 ...

  6. Cobalt Strike简单使用(9,29第十五天)

    本文转自:https://www.cnblogs.com/yuanshu/p/11616657.html 一.介绍: 后渗透测试工具,基于Java开发,适用于团队间协同作战,简称“CS”. CS分为客 ...

  7. EUI库 - 快速入口之项目配置

      egretProperties.json exmlRoot  指定exml文件存放根目录,该路径必须为相对路径,目录内只能有exml文件 themes    主题文件数组,配置所有主题文件路径,该 ...

  8. 创建简单web项目

    Intellij Idea直接安装(可根据需要选择自己设置的安装目录),jdk使用1.6/1.7/1.8都可以,主要是配置好系统环境变量,tomcat7上tomcat的官网下载压缩包解压即可. 一.创 ...

  9. Hibernate 的SessionFactory

    1.当我们调用 Configuration config=new Configuration().configure(); 时候Hibernate会自动在当前的CLASSPATH中搜寻hibernat ...

  10. PWC6199:Generated servlet error:Only a type can be imported. org.apache.jasper.tagplugins.jstl.core.ForEach resolves to a package

    <%@ import="org.apache.jasper.tagplugins.jstl.core.ForEach"%> 去掉这条语句,就不报错了.所以问题就出在这里 ...