今天就跟大家聊聊有關(guān)Go編寫(xiě)的Socket服務(wù)器模塊解耦及基礎(chǔ)模塊的設(shè)計(jì)示例分析,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。
為文圣等地區(qū)用戶提供了全套網(wǎng)頁(yè)設(shè)計(jì)制作服務(wù),及文圣網(wǎng)站建設(shè)行業(yè)解決方案。主營(yíng)業(yè)務(wù)為成都網(wǎng)站設(shè)計(jì)、成都網(wǎng)站建設(shè)、外貿(mào)網(wǎng)站建設(shè)、文圣網(wǎng)站設(shè)計(jì),以傳統(tǒng)方式定制建設(shè)網(wǎng)站,并提供域名空間備案等一條龍服務(wù),秉承以專(zhuān)業(yè)、用心的態(tài)度為用戶提供真誠(chéng)的服務(wù)。我們深信只要達(dá)到每一位用戶的要求,就會(huì)得到認(rèn)可,從而選擇與我們長(zhǎng)期合作。這樣,我們也可以走得更遠(yuǎn)!Server的解耦—通過(guò)Router+Controller實(shí)現(xiàn)邏輯分發(fā)
在實(shí)際的系統(tǒng)項(xiàng)目工程中中,我們?cè)趯?xiě)代碼的時(shí)候要盡量避免不必要的耦合,否則你以后在更新和維護(hù)代碼的時(shí)候會(huì)發(fā)現(xiàn)如同深陷泥潭,隨便改點(diǎn)東西整個(gè)系統(tǒng)都要變動(dòng)的酸爽會(huì)讓你深切后悔自己當(dāng)初為什么非要把東西都寫(xiě)到一塊去(我不會(huì)說(shuō)我剛實(shí)習(xí)的時(shí)候就是這么干的。。。)
所以這一篇主要說(shuō)說(shuō)如何設(shè)計(jì)Sever的內(nèi)部邏輯,將Server處理Client發(fā)送信息的這部分邏輯與Sevrer處理Socket連接的邏輯進(jìn)行解耦~
這一塊的實(shí)現(xiàn)靈感主要是在讀一個(gè)HTTP開(kāi)源框架: Beego 的源代碼的時(shí)候產(chǎn)生的,Beego的整個(gè)架構(gòu)就是高度解耦的,這里引用一下作者的介紹:
beego 是基于八大獨(dú)立的模塊構(gòu)建的,是一個(gè)高度解耦的框架。當(dāng)初設(shè)計(jì) beego 的時(shí)候就是考慮功能模塊化,用戶即使不使用 beego 的 HTTP 邏輯,也依舊可以使用這些獨(dú)立模塊,例如:你可以使用 cache 模塊來(lái)做你的緩存邏輯;使用日志模塊來(lái)記錄你的操作信息;使用 config 模塊來(lái)解析你各種格式的文件。所以 beego 不僅可以用于 HTTP 類(lèi)的應(yīng)用開(kāi)發(fā),在你的 socket 游戲開(kāi)發(fā)中也是很有用的模塊,這也是 beego 為什么受歡迎的一個(gè)原因。大家如果玩過(guò)樂(lè)高的話,應(yīng)該知道很多高級(jí)的東西都是一塊一塊的積木搭建出來(lái)的,而設(shè)計(jì) beego 的時(shí)候,這些模塊就是積木,高級(jí)機(jī)器人就是 beego。
這里上一張Beego的架構(gòu)圖:
這是一個(gè)典型的MVC框架,可以看到,當(dāng)用戶發(fā)送請(qǐng)求到beego后,Beego內(nèi)部在通過(guò)路由進(jìn)行參數(shù)的過(guò)濾,然后路由根據(jù)用戶發(fā)來(lái)的參數(shù)判斷調(diào)用哪個(gè)Controller執(zhí)行相關(guān)的邏輯,并在controller里調(diào)用相關(guān)的模塊實(shí)現(xiàn)功能。通過(guò)這種方式,Beego成功的將所有模塊都獨(dú)立出來(lái),也就是astaxie所說(shuō)的“樂(lè)高積木化”。
在這里,我們可以仿照Beego的架構(gòu),在Server內(nèi)部加入一層Router,通過(guò)Router對(duì)通過(guò)Socket發(fā)來(lái)的信息進(jìn)通過(guò)我們?cè)O(shè)定的規(guī)則行的判斷后,調(diào)用相關(guān)的Controller進(jìn)行任務(wù)的分發(fā)處理。在這個(gè)過(guò)程中不僅Controller彼此獨(dú)立,匹配規(guī)則和Controller之間也是相互獨(dú)立的。
下面給出Router的實(shí)現(xiàn)代碼,其中Msg的結(jié)構(gòu)對(duì)應(yīng)的是Json字符串,當(dāng)然考慮到實(shí)習(xí)公司現(xiàn)在也在用這個(gè),修改了一部分,不過(guò)核心思路是一樣的哦:
復(fù)制代碼 代碼如下:
import (
"utils"
"fmt"
"encoding/json"
)
type Msg struct {
Conditions map[string]interface{} `json:"meta"`
Content interface{} `json:"content"`
}
type Controller interface {
Excute(message Msg) []byte
}
var routers [][2]interface{}
func Route(judge interface{} ,controller Controller) {
switch judge.(type) {
case func(entry Msg)bool:{
var arr [2]interface{}
arr[0] = judge
arr[1] = controller
routers = append(routers,arr)
}
case map[string]interface{}:{
defaultJudge:= func(entry Msg)bool{
for keyjudge , valjudge := range judge.(map[string]interface{}){
val, ok := entry.Meta[keyjudge]
if !ok {
return false
}
if val != valjudge {
return false
}
}
return true
}
var arr [2]interface{}
arr[0] = defaultjudge
arr[1] = controller
routers = append(routers,arr)
fmt.Println(routers)
}
default:
fmt.Println("Something is wrong in Router")
}
}
通過(guò)自定義接口Router,我們將匹配規(guī)則judge和對(duì)應(yīng)的controller封裝了進(jìn)去,然后在Server端負(fù)責(zé)接收socket發(fā)送信息的函數(shù)handleConnection那里再實(shí)現(xiàn)Router內(nèi)部的遍歷即可:
復(fù)制代碼 代碼如下:
for _ ,v := range routers{
pred := v[0]
act := v[1]
var message Msg
err := json.Unmarshal(postdata,&message)
if err != nil {
Log(err)
}
if pred.(func(entry Msg)bool)(message) {
result := act.(Controller).Excute(message)
conn.Write(result)
return
}
}
這樣Client每次發(fā)來(lái)信息,我們就可以讓Router自動(dòng)跟現(xiàn)有的規(guī)則進(jìn)行匹配,最后調(diào)用對(duì)應(yīng)的Controller進(jìn)行邏輯的實(shí)現(xiàn)啦,下面給出一個(gè)controller的編寫(xiě)實(shí)例,這個(gè)Controll的作用是發(fā)來(lái)的json類(lèi)型是mirror的時(shí)候,將Client發(fā)來(lái)的信息原樣返回:
復(fù)制代碼 代碼如下:
type MirrorController struct {
}
func (this *MirrorController) Excute(message Msg)[]byte {
mirrormsg,err :=json.Marshal(message)
CheckError(err)
return mirrormsg
}
func init() {
var mirror
routers = make([][2]interface{} ,0 , 20)
Route(func(entry Msg)bool{
if entry.Meta["msgtype"]=="mirror"{
return true}
return false
},&mirror)
}
日志模塊的設(shè)計(jì)與定時(shí)任務(wù)模塊模塊
作為一個(gè)Server,日志(Log)功能是必不可少的,一個(gè)設(shè)計(jì)良好的日志模塊,不論是開(kāi)發(fā)Server時(shí)的調(diào)試,還是運(yùn)行時(shí)候的維護(hù),都是非常有幫助的。
因?yàn)檫@里寫(xiě)的是一個(gè)比較簡(jiǎn)化的Server框架,因此我選擇對(duì)Golang本身的log庫(kù)進(jìn)行擴(kuò)充,從而實(shí)現(xiàn)一個(gè)簡(jiǎn)單的Log模塊。
在這里,我將日志的等級(jí)大致分為Debug,Operating,Error 3個(gè)等級(jí),Debug主要用于存放調(diào)試階段的日志信息,Operateing用于保存Server日常運(yùn)行時(shí)產(chǎn)生的信息,Error則是保存報(bào)錯(cuò)信息。
模塊代碼如下:
復(fù)制代碼 代碼如下:
func LogErr(v ...interface{}) {
logfile := os.Stdout
log.Println(v...)
logger := log.New(logfile,"\r\n",log.Llongfile|log.Ldate|log.Ltime);
logger.SetPrefix("[Error]")
logger.Println(v...)
defer logfile.Close();
}
func Log(v ...interface{}) {
logfile := os.Stdout
log.Println(v...)
logger := log.New(logfile,"\r\n",log.Ldate|log.Ltime);
logger.SetPrefix("[Info]")
logger.Println(v...)
defer logfile.Close();
}
func LogDebug(v ...interface{}) {
logfile := os.Stdout
log.Println(v...)
logger := log.New(logfile,"\r\n",log.Ldate|log.Ltime);
logger.SetPrefix("[Debug]")
logger.Println(v...)
defer logfile.Close();
}
func CheckError(err error) {
if err != nil {
LogErr(os.Stderr, "Fatal error: %s", err.Error())
}
}
注意這里log的輸出我使用的是stdout,因?yàn)檫@樣在Server運(yùn)行的時(shí)候可以直接將log重定向到指定的位置,方便整個(gè)Server的部署。不過(guò)在日常開(kāi)發(fā)的時(shí)候,為了方便調(diào)試代碼,我推薦將log輸出到指定文件位置下,這樣在調(diào)試的時(shí)候會(huì)方便很多(主要是因?yàn)間olang的調(diào)試實(shí)在太麻煩,很多時(shí)候都要依靠打log的時(shí)候進(jìn)行步進(jìn)。便于調(diào)試的Log模塊代碼示意:
復(fù)制代碼 代碼如下:
func Log(v ...interface{}) {
logfile := os.OpenFile("server.log",os.O_RDWR|os.O_APPEND|os.O_CREATE,0);
if err != nil {
fmt.Fprintf(os.Stderr, "Fatal error: %s", err.Error())
return }
log.Println(v...)
logger := log.New(logfile,"\r\n",log.Ldate|log.Ltime);
logger.SetPrefix("[Info]")
logger.Println(v...)
defer logfile.Close();
}
然后就是計(jì)時(shí)循環(huán)模塊啦,日常運(yùn)行中,Server經(jīng)常要執(zhí)行一些定時(shí)任務(wù),比如隔一定時(shí)間刷新后臺(tái),隔一段時(shí)間自動(dòng)刷新爬蟲(chóng)等等,在這里我設(shè)計(jì)了一個(gè)Task接口,通過(guò)類(lèi)似于TaskList的的方式將所有定時(shí)任務(wù)注冊(cè)后統(tǒng)一執(zhí)行,代碼如下:
復(fù)制代碼 代碼如下:
type DoTask interface {
Excute()
}
var tasklist []interface{}
func AddTask(controller DoTask) {
var arr interface{}
arr = controller
tasklist = append(tasklist,arr)
fmt.Println(tasklist)
}
在這里以一個(gè)定時(shí)報(bào)時(shí)任務(wù)作為例子:
復(fù)制代碼 代碼如下:
type Task1 struct {}
func (this * Task1)Excute() {
timer := time.NewTicker(2 * time.Second)
for {
select {
case <-timer.C:
go func() {
Log(time.Now())
}()
}
}
}
func init() {
var task1 Task1
tasklist = make([]interface{} ,0 , 20)
AddTask(&task1)
for _, v := range tasklist {
v.(DoTask).Excute()
}
}
注意這里的定時(shí)任務(wù)要做成非阻塞的,否則整個(gè)Server都會(huì)卡在tasklist的第一個(gè)task的。。。
看完上述內(nèi)容,你們對(duì)Go編寫(xiě)的Socket服務(wù)器模塊解耦及基礎(chǔ)模塊的設(shè)計(jì)示例分析有進(jìn)一步的了解嗎?如果還想了解更多知識(shí)或者相關(guān)內(nèi)容,請(qǐng)關(guān)注創(chuàng)新互聯(lián)成都網(wǎng)站設(shè)計(jì)公司行業(yè)資訊頻道,感謝大家的支持。
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無(wú)理由+7*72小時(shí)售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國(guó)服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡(jiǎn)單易用、服務(wù)可用性高、性價(jià)比高”等特點(diǎn)與優(yōu)勢(shì),專(zhuān)為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場(chǎng)景需求。