Go语言-并发模式-资源池实例(pool)
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)的更多相关文章
- Go语言-并发模式-goroutine池实例(work)
介绍 使用无缓冲的通道来创建一个 goroutine 池,这些 goroutine 执行并控制一组工作,让其并发执行.在这种情况下,使用无缓冲的通道要比随意指定一个缓冲区大小的有缓冲的通道好,因为这个 ...
- 《Go语言实战》摘录:7.2 并发模式 - pool
7.2 并发模式 - pool
- WCF实例上下文模式与并发模式对性能的影响
实例上下文模式 InstanceContextMode 控制在响应客户端调用时,如何分配服务实例.InstanceContextMode 可以设置为以下值: •Single – 为所有客户端调用分配一 ...
- go语言】Goroutines 并发模式
并发模式 让我们先来回顾一下boring函数的例子. func boring(msg string, c chan string) { for i := 0; ; i++ { c ...
- 《Go语言实战》摘录:7.3 并发模式 - work
7.3 并发模式 - work
- 《Go语言实战》摘录:7.1 并发模式 - runner
7.1 并发模式 - runner
- 《C#并发编程经典实例》笔记
1.前言 2.开宗明义 3.开发原则和要点 (1)并发编程概述 (2)异步编程基础 (3)并行开发的基础 (4)测试技巧 (5)集合 (6)函数式OOP (7)同步 1.前言 最近趁着项目的一段平稳期 ...
- Go并发模式:管道与取消
关键字:Go语言,管道,取消机制,并发,sync.WaitGroup,包引用,通道,defer,select GO并发模式:管道与取消 简介 Go的并发能力可以使构建一个流数据管道变得非常容易,并且可 ...
- 09. Go 语言并发
Go 语言并发 并发指在同一时间内可以执行多个任务.并发编程含义比较广泛,包含多线程编程.多进程编程及分布式程序等.本章讲解的并发含义属于多线程编程. Go 语言通过编译器运行时(runtime),从 ...
随机推荐
- 吴裕雄--天生自然java开发常用类库学习笔记:Math与Random类
public class MathDemo01{ public static void main(String args[]){ // Math类中的方法都是静态方法,直接使用“类.方法名称()”的形 ...
- B. Sport Mafia 二分
B. Sport Mafia time limit per test 2 seconds memory limit per test 256 megabytes input standard inpu ...
- redis学习(三)
一.Redis 数据类型 Redis支持五种数据类型:string(字符串),hash(哈希),list(列表),set(集合)及zset(sorted set:有序集合). 二.Redis 命令 1 ...
- Charles抓包(HTTP)
一.电脑抓包: 安装Charles,打开Charles即可 二.手机抓包: 设置手机WiFi配置代理即可:(确保电脑和手机在同一个网络) 三.拦截请求: 四.修改请求/返回: 打上断点后,刷新页面,在 ...
- C语言备忘录——static
对于这个关键字我一直没有弄清楚,今天特地去花了一定的时间去理解这个关键字.在函数或变量声明时,在数据类型前加上 static 后会有以下几个效果 一.用于函数定义时: 1.函数的链接属性会被修改,从e ...
- HDU - 6205 card card card (尺取法)
题意:有n堆牌,ai表示每堆牌的牌数,bi表示每堆牌的penaltyvalue,操作开始前,可以重复进行将第一堆牌挪到最后一堆这一操作.然后,对于挪完后的牌,从第一堆开始,依次取.对于每一堆牌,首先将 ...
- netty权威指南学习笔记一——NIO入门(4)AIO
NIO2.0引入了新的异步通道的概念,并提供了异步文件通道和异步套接字通道的实现.异步通道提供以下两种方式获取操作结果. 1.通过java.util.concurrent.Future 类来表示异步操 ...
- cf 762C. Two strings
因为要删去1个串(读错题),所以就直接二分搞就好了. 需要预处理出2个分别从头到尾,或从尾到头需要多长a串的数组,然后二分删去多长就好了. #include<bits/stdc++.h> ...
- 开源DDD设计模式框架YMNNetCoreFrameWork第四篇-增加YMNSession,增加异常处理,增加策略授权
1.增加YMNSession,可以获取当前登录的用户信息 2.增加异常处理,不要使用过滤器,过滤器只能获取到mvc异常 3.增加策略授权,策略授权支持用户角色.用户名授权,初步实现 代码地址:http ...
- 八、JavaScript之执行语句
一.代码如下 二.运行结果如下 <!DOCTYPE html> <html> <meta http-equiv="Content-Type" cont ...