首先要說明的是,這裡實現的異步推送服務采用的是Long Polling方式,並不是Comet。
如果想用Comet來實現的話,可以參考這個開源項目:http://cometd.org/。不過其中的服務端實現只有Java版和Python版。如果要用Go來做後端的話需要自己實現Bayeux協議。
關於異步推送服務的解決方案的資料有很多,在這裡就不在贅述了。當然,當前最先進的兩個方案就是Long Polling和Comet。
1. 預備知識
1.1 Go語言
關於Go語言,其實要說的很多。但是為了不跑題,請大家移步到這裡:http://code.google.com/p/golang-china/。另外,Go語言的官網地址是:http://golang.org/。
1.2 Hijack
Hijack其實是一個單詞,雖然有很多人把它和電影《泰坦尼克號》中Rose的召喚聯系到一起。Hijack被譯為劫持,在“處理HTTP請求”的這個上下文中,就意味著可以讓我們“劫持”(或者說“保持”)HTTP請求鏈接,做一些其他操作(比如根據需要修改HTTP響應的內容),然後再在之後的某個時間將響應“推送”回去。說到這,我想這就與Long Polling的運作方式很相似了。
Go語言的Hijack接口非常簡單,我們在官方的文檔站點上可以找到說明:http://godoc.org/net/http#Hijacker。本文中的核心代碼也是來自於此文檔。
1.3 jQuery
jQuery作為當今最流行的Javascript開發框架,我想基本上每一個做過Web開發的人都會知道,所以在這裡我就不多說了。如果你不知道,可以看這裡:http://jquery.com/。
2. 實戰
2.1 需求
在本案例中,我需要做一個能實時查看當前授權碼的頁面,而且我不想手動刷新頁面。另外,我還想記錄一下從頁面打開到當前時刻授權碼改變過多少次。因為授權碼在被使用後會自動變更一次,所以授權的變更次數就等於使用授權碼服務的人數。
2.2 編寫服務端代碼
之前說了,我們使用Go語言來編寫後端代碼。我們要使用Go語言的官方http庫。
其中,我們需要用http.HandleFunc來注冊針對某個url的請求處理器。如下:
http.HandleFunc("/auth_code", getAuthCodeForAdmin) //向http服務器注冊一個對指定url進行處理的函數。
在函數getAuthCodeForAdmin的簽名中,有兩個參數——http.Request對象指針和http.ResponseWriter對象。http.Request對象指針用來獲取請求信息,http.ResponseWriter對象用來寫入響應。
如果要使用Go的Hijack方式來處理HTTP請求,就需要先import其官方的http包:
import (
"net/http"
)
之後,我們在處理函數getAuthCodeForAdmin中先將http.ResponseWriter對象顯式轉換為http.Hijacker接口:
hj, ok := w.(http.Hijacker)
返回值中賦給“ok”變量的值代表轉換是否成功,如果不成功就說明http.ResponseWriter對象未實現http.Hijacker接口。
如果轉換成功,我們就可以調用http.Hijacker接口的Hijack方法來獲取連接對象及其讀寫緩存對象了:
conn, bufrw, err := hj.Hijack()
返回值中,“conn”代表連接對象,“bufrw”代表該連接的讀寫緩存對象。
如果返回值“err”等於nil就說明獲取成功,我就可以繼續下面的事情了。但首先需要在函數推出前關閉連接,不論函數是否執行成功以及是否有錯誤發生:
defer conn.Close()
使用defer關鍵字意味著,讓程序執行流程退出該函數前先執行緊隨其後的語句或函數。這樣就保證了資源的及時釋放。
接下來,我們先觀測新的授權碼的出現,當其出現後我們就使用連接讀寫緩存對象bufrw返回給http客戶端。從觀測到返回給http客戶端的時間並不確定,也許時間會很長,這也從側面體現了Long Polling中的Long。看下面的代碼:
nacChan := make(chan string)
triggerFunc := func(newAuthCode string) {
nacChan <- newAuthCode }
triggerId := fmt.Sprintf("long-polling|%s|%s|%d", loginName, groupName, time.Now().UnixNano())
request.AddNewAuthCodeTrigger(triggerId, triggerFunc)
defer request.DelNewAuthCodeTrigger(triggerId)
這段代碼其中包含的東西很多,我們不需要全搞明白,只要知道這是為新授權碼產生時間注冊一個觸發器就行了。
當新授權碼被產生後,充當觸發器的函數triggerFunc會被調用。它會向名為nacChan的Channel中添加一個元素。注意,這個Channel是字符串類型的,並且是阻塞式。阻塞式意味著獲取元素的語句會一直阻塞,直到該Channel被添入元素。另外,當Channel中已有一個元素時,添加元素的語句也會被阻塞。我們在這裡只用到了阻塞式Channel的前一個特性。元素獲取語句是這樣寫的:
newAuthCode := <-nacChan
獲取到新授權碼後,程序會立即把它“push”給http客戶端。
done := pushResponse(bufrw, newAuthCode)
函數pushResponse的完整定義如下:
func pushResponse(bufrw *bufio.ReadWriter, authCode string) bool {
_, err := bufrw.Write([]byte(authCode))
if err == nil {
err = bufrw.Flush()
}
if err != nil {
go_lib.LogErrorf("PushAuthCodeError: %s\n", err)
return false
}
return true
}
其中用到了很多“net/http”以外的包,關於它們的說明可以到Go文檔站點http://godoc.org/中查找。另外,“go_lib”是我為了自己開發方便而寫的一個函數庫,源碼在這裡:https://github.com/hyper-carrot/go_lib,有興趣的讀者可以查看。
至此,基於Long Polling的異步推送服務的服務端就完成了。函數getAuthCodeForAdmin的完整代碼可以參看:https://github.com/hyper-carrot/hypermind/blob/master/server.go#L244。