|
@@ -7,14 +7,15 @@ import (
|
|
|
"syscall"
|
|
|
)
|
|
|
|
|
|
-// 工作对象封装
|
|
|
+// chan worker object
|
|
|
type ChanWorker struct {
|
|
|
- ID int // 工作对象编号
|
|
|
- WorkerPool chan chan interface{} // 工作管道池,实例化时由调度器传入
|
|
|
- JobChannel chan interface{} // 工作管道
|
|
|
- quit chan bool // 退出消息
|
|
|
+ ID int // worker ID
|
|
|
+ WorkerPool chan chan interface{} // refer to chan worker pool
|
|
|
+ JobChannel chan interface{} // job message chan
|
|
|
+ quit chan bool // quit chan
|
|
|
}
|
|
|
|
|
|
+// NewChanWorker() get new chan worker object instance
|
|
|
func NewChanWorker(workerId int, workerPool chan chan interface{}) *ChanWorker {
|
|
|
jobChannel := make(chan interface{})
|
|
|
|
|
@@ -26,15 +27,16 @@ func NewChanWorker(workerId int, workerPool chan chan interface{}) *ChanWorker {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+// Start() worker handling
|
|
|
func (w *ChanWorker) Start(callback func(workerId int, msg interface{})) {
|
|
|
go func(w *ChanWorker, callback func(workerId int, msg interface{})) {
|
|
|
- defer utils.DefaultGoroutineRecover(nil, `chan池工作对象消息处理`)
|
|
|
+ defer utils.DefaultGoroutineRecover(nil, `chan pool worker handling`)
|
|
|
for {
|
|
|
- // 新工作管道或每次取用工作管道后,加入工作管道池
|
|
|
+ // job chan add to pool,when create or used
|
|
|
w.WorkerPool <- w.JobChannel
|
|
|
|
|
|
select {
|
|
|
- case msg, ok := <-w.JobChannel: // 无消息时阻塞
|
|
|
+ case msg, ok := <-w.JobChannel: // block when empty
|
|
|
if ok {
|
|
|
callback(w.ID, msg)
|
|
|
}
|
|
@@ -45,14 +47,15 @@ func (w *ChanWorker) Start(callback func(workerId int, msg interface{})) {
|
|
|
}(w, callback)
|
|
|
}
|
|
|
|
|
|
-// 调度对象
|
|
|
+// chan worker pool dispatcher object
|
|
|
type ChanDispatcher struct {
|
|
|
- MsgQueue chan interface{} // 消息输入管道
|
|
|
- WorkerPool chan chan interface{} // 工作管道池
|
|
|
- maxWorkers int // 最大工作对象数
|
|
|
- workers []*ChanWorker // 工作对象列表
|
|
|
+ MsgQueue chan interface{} // message input chan
|
|
|
+ WorkerPool chan chan interface{} // worker chan pool
|
|
|
+ maxWorkers int // max concurrent chan count
|
|
|
+ workers []*ChanWorker // chan worker object list
|
|
|
}
|
|
|
|
|
|
+// NewChanDispatcher() get new chan worker pool dispatcher object instance
|
|
|
func NewChanDispatcher(msgQueue chan interface{}, maxWorkers int) *ChanDispatcher {
|
|
|
return &ChanDispatcher{
|
|
|
MsgQueue: msgQueue,
|
|
@@ -61,6 +64,7 @@ func NewChanDispatcher(msgQueue chan interface{}, maxWorkers int) *ChanDispatche
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+// Run() start concurrent message customers base on chan worker pool dispatcher
|
|
|
func (d *ChanDispatcher) Run(callback func(workerId int, msg interface{})) {
|
|
|
for i := 0; i < d.maxWorkers; i++ {
|
|
|
worker := NewChanWorker(i, d.WorkerPool)
|
|
@@ -71,18 +75,20 @@ func (d *ChanDispatcher) Run(callback func(workerId int, msg interface{})) {
|
|
|
d.dispatch()
|
|
|
}
|
|
|
|
|
|
+// dispatch() dispatch handling
|
|
|
func (d *ChanDispatcher) dispatch() {
|
|
|
go func(d *ChanDispatcher) {
|
|
|
- defer utils.DefaultGoroutineRecover(nil, `chan池调度`)
|
|
|
+ defer utils.DefaultGoroutineRecover(nil, `chan pool dispatch handling`)
|
|
|
for {
|
|
|
select {
|
|
|
case msg, ok := <-d.MsgQueue:
|
|
|
if ok {
|
|
|
- // 从工作管道池中尝试取出一个空闲的工作管道(每次取用工作管道会从池中取出去,消息处理完再放回池子,所以池子中的都是空闲的)
|
|
|
- // 无空闲工作管道(池子中无消息)时阻塞
|
|
|
+ // try to get a free worker chan from pool
|
|
|
+ // (all worker chan in pool is free,because take it out when use,and put it back when used)
|
|
|
+ // block when all worker channels are busy(pool is empty)
|
|
|
jobChannel, isOpen := <-d.WorkerPool
|
|
|
if isOpen {
|
|
|
- // 将一条消息发送给成功取出的工作管道
|
|
|
+ // send a message to job chan
|
|
|
jobChannel <- msg
|
|
|
}
|
|
|
}
|
|
@@ -91,6 +97,7 @@ func (d *ChanDispatcher) dispatch() {
|
|
|
}(d)
|
|
|
}
|
|
|
|
|
|
+// closeWait() release dispatcher resource when signal close
|
|
|
func (d *ChanDispatcher) closeWait() {
|
|
|
go func(d *ChanDispatcher) {
|
|
|
defer utils.DefaultGoroutineRecover(nil, `chan池关闭`)
|