go 實現(xiàn)一個簡易的線程池(二)

針對(一)中所出現(xiàn)的問題荸百,現(xiàn)在使用一個兩級channel系統(tǒng),一個用來存放任務(wù)隊列株茶,另一個用來控制任務(wù)隊列上執(zhí)行操作的"工人"數(shù)量([參考文章]:(http://marcio.io/2015/07/handling-1-million-requests-per-minute-with-golang/))目代。

  • 先將具體的業(yè)務(wù)操作抽取出來,里面定義一個具體的業(yè)務(wù)方法
// MyService 業(yè)務(wù)接口
type MyService struct {
}

// WriteInfo 寫日志
func (s *MyService) WriteInfo() {
    time.Sleep(1 * time.Second)
    t := time.Now()
    logFile, err := os.OpenFile("syslog.txt", os.O_RDWR|os.O_CREATE|os.O_APPEND, 0766)
    defer logFile.Close()
    if err != nil {
        panic(err)
    }
    infoLog := log.New(logFile, "[INFO]", log.LstdFlags)
    infoLog.Print("time=" + strconv.FormatInt(t.UTC().UnixNano(), 10))
}
  • 定義一個代表工作的結(jié)構(gòu)體Job惕味,和一個代表工人的結(jié)構(gòu)體Worker
// Job 表示要執(zhí)行的作業(yè)
type Job struct {
    MyService MyService
}

// Worker 執(zhí)行作業(yè)的工人
type Worker struct {
    WorkerPool chan chan Job
    JobChannel chan Job
}
// NewWorker 新建一個工人
func NewWorker(workerPool chan chan Job) Worker {
    return Worker{
        WorkerPool: workerPool,
        JobChannel: make(chan Job),
    }
}
  • 給工人定義一個Start方法,表示監(jiān)聽自己的工作任務(wù)矢劲,有活兒來了就開始工作
func (w Worker) Start() {
    go func() {
        for {
            w.WorkerPool <- w.JobChannel
            select {
            case job := <-w.JobChannel:
                //有工作任務(wù)時赦拘,開始執(zhí)行業(yè)務(wù)接口的方法
                job.MyService.WriteInfo()
            }
        }
    }()
}
  • 初始化池
var maxWorkers = 20

// JobQueue 作業(yè)隊列
var JobQueue = make(chan Job, maxWorkers)

// InitPool 給池中初始化一定量的工人,以及開啟任務(wù)隊列的監(jiān)聽
func InitPool() {
    // 創(chuàng)建工作池
    pool := make(chan chan Job, maxWorkers)
    // 創(chuàng)建一定數(shù)量的工人(可以看做:創(chuàng)建了N個工人芬沉,每個工人能并發(fā)處理N件工作)
    for i := 0; i < maxWorkers; i++ {
        worker := NewWorker(pool)
        worker.Start()
    }
    //監(jiān)聽JobQueue上是否有新任務(wù)
    go func() {
        for {
            select {
            case job := <-JobQueue:
                go func(job Job) {
                    // 獲取可用的工人channel躺同,若沒有,則阻塞
                    jobChannel := <-pool
                    jobChannel <- job
                }(job)
            }
        }
    }()
}
  • 提供一個http接口丸逸,用于測試
func init() {
    InitPool()
}

func main() {
    http.HandleFunc("/test/pool/", indexHandler)
    http.ListenAndServe(":9000", nil)
}

func indexHandler(w http.ResponseWriter, r *http.Request) {
    service := MyService{}
    //service.WriteInfo()
    work := Job{MyService: service}
    JobQueue <- work
    w.Header().Set("Content-Type", "application/json; charset=UTF-8")
    result := "{\"msg\":\"SUCCESS\",\"code\":0}"
    fmt.Fprintln(w, result)
}
  • 使用jmeter進行測試蹋艺,和(一)中一樣,100個樣本黄刚,循環(huán)10次執(zhí)行


    聚合報告.png
圖形結(jié)果.png

可以看到處理請求的能力還是非常不錯的捎谨。其實這種測試方法并不能和(一)的結(jié)果進行橫向比對,畢竟(一)中只開啟了20個協(xié)程憔维,而在(二)中由于雙隊列的存在涛救,處理請求的協(xié)程數(shù)量肯定是要比(一)中的多的多。
但(二)的模式肯定是要優(yōu)于(一)的业扒,這是毋容置疑的检吆。我只是為了給自己做個筆記,加深對go中channel的理解程储,畢竟蹭沛,好記性不如爛筆頭么 (? ??_??)?
源碼我上傳到了github,地址:https://github.com/wleirock/studygo/tree/master/pool

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末章鲤,一起剝皮案震驚了整個濱河市摊灭,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌败徊,老刑警劉巖帚呼,帶你破解...
    沈念sama閱讀 211,123評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異集嵌,居然都是意外死亡萝挤,警方通過查閱死者的電腦和手機御毅,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,031評論 2 384
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來怜珍,“玉大人端蛆,你說我怎么就攤上這事∷址海” “怎么了今豆?”我有些...
    開封第一講書人閱讀 156,723評論 0 345
  • 文/不壞的土叔 我叫張陵,是天一觀的道長柔袁。 經(jīng)常有香客問我呆躲,道長,這世上最難降的妖魔是什么捶索? 我笑而不...
    開封第一講書人閱讀 56,357評論 1 283
  • 正文 為了忘掉前任插掂,我火速辦了婚禮,結(jié)果婚禮上腥例,老公的妹妹穿的比我還像新娘辅甥。我一直安慰自己,他們只是感情好燎竖,可當(dāng)我...
    茶點故事閱讀 65,412評論 5 384
  • 文/花漫 我一把揭開白布璃弄。 她就那樣靜靜地躺著,像睡著了一般构回。 火紅的嫁衣襯著肌膚如雪夏块。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,760評論 1 289
  • 那天纤掸,我揣著相機與錄音脐供,去河邊找鬼。 笑死借跪,一個胖子當(dāng)著我的面吹牛患民,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播垦梆,決...
    沈念sama閱讀 38,904評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼仅孩!你這毒婦竟也來了托猩?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,672評論 0 266
  • 序言:老撾萬榮一對情侶失蹤辽慕,失蹤者是張志新(化名)和其女友劉穎京腥,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體溅蛉,經(jīng)...
    沈念sama閱讀 44,118評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡公浪,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,456評論 2 325
  • 正文 我和宋清朗相戀三年他宛,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片欠气。...
    茶點故事閱讀 38,599評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡厅各,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出预柒,到底是詐尸還是另有隱情队塘,我是刑警寧澤,帶...
    沈念sama閱讀 34,264評論 4 328
  • 正文 年R本政府宣布宜鸯,位于F島的核電站憔古,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏淋袖。R本人自食惡果不足惜鸿市,卻給世界環(huán)境...
    茶點故事閱讀 39,857評論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望即碗。 院中可真熱鬧焰情,春花似錦、人聲如沸拜姿。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,731評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽蕊肥。三九已至谒获,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間壁却,已是汗流浹背批狱。 一陣腳步聲響...
    開封第一講書人閱讀 31,956評論 1 264
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留展东,地道東北人赔硫。 一個月前我還...
    沈念sama閱讀 46,286評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像盐肃,于是被迫代替她去往敵國和親爪膊。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,465評論 2 348

推薦閱讀更多精彩內(nèi)容