青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

Fork me on GitHub
隨筆 - 215  文章 - 13  trackbacks - 0
<2017年1月>
25262728293031
1234567
891011121314
15161718192021
22232425262728
2930311234


專注即時(shí)通訊及網(wǎng)游服務(wù)端編程
------------------------------------
Openresty 官方模塊
Openresty 標(biāo)準(zhǔn)模塊(Opm)
Openresty 三方模塊
------------------------------------
本博收藏大部分文章為轉(zhuǎn)載,并在文章開(kāi)頭給出了原文出處,如有再轉(zhuǎn),敬請(qǐng)保留相關(guān)信息,這是大家對(duì)原創(chuàng)作者勞動(dòng)成果的自覺(jué)尊重!!如為您帶來(lái)不便,請(qǐng)于本博下留言,謝謝配合。

常用鏈接

留言簿(1)

隨筆分類

隨筆檔案

相冊(cè)

Awesome

Blog

Book

GitHub

Link

搜索

  •  

積分與排名

  • 積分 - 219730
  • 排名 - 117

最新評(píng)論

閱讀排行榜

http://www.jb51.net/article/80771.htm

Server的解耦—通過(guò)Router+Controller實(shí)現(xiàn)邏輯分發(fā)

在實(shí)際的系統(tǒng)項(xiàng)目工程中中,我們?cè)趯懘a的時(shí)候要盡量避免不必要的耦合,否則你以后在更新和維護(hù)代碼的時(shí)候會(huì)發(fā)現(xiàn)如同深陷泥潭,隨便改點(diǎn)東西整個(gè)系統(tǒng)都要變動(dòng)的酸爽會(huì)讓你深切后悔自己當(dāng)初為什么非要把東西都寫到一塊去(我不會(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 類的應(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)圖:

2016310160604529.png (793×291)


 

  這是一個(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的編寫實(shí)例,這個(gè)Controll的作用是發(fā)來(lái)的json類型是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)檫@里寫的是一個(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ò)類似于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的。。。

 

 

您可能感興趣的文章:

posted on 2016-08-23 16:36 思月行云 閱讀(163) 評(píng)論(0)  編輯 收藏 引用 所屬分類: Golang
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            午夜精品久久久久久99热软件| 亚洲视频一二三| 久久免费精品视频| 久久久久久黄| 欧美va亚洲va国产综合| 欧美电影资源| 欧美性猛交视频| 国产亚洲欧洲| 亚洲日本黄色| 亚洲欧美资源在线| 欧美视频在线观看| 国产精品午夜春色av| 国产真实乱偷精品视频免| 黑人一区二区| 99亚洲视频| 久久久久久黄| 亚洲激情视频在线观看| 99日韩精品| 久久久xxx| 欧美午夜精品久久久| 国产在线欧美| 99国产精品自拍| 久久久久久69| 99在线|亚洲一区二区| 欧美伊人精品成人久久综合97 | 亚洲综合三区| 免费欧美视频| 亚洲女性喷水在线观看一区| 免费观看国产成人| 国产日韩欧美黄色| 一区二区电影免费观看| 久久一区二区精品| 中文亚洲欧美| 欧美精品久久99| 在线看片日韩| 久久se精品一区二区| 日韩亚洲不卡在线| 免费观看30秒视频久久| 国产在线精品二区| 午夜精品一区二区三区电影天堂 | 欧美日韩在线视频一区| 亚洲电影第三页| 久久精品av麻豆的观看方式| 日韩一区二区福利| 欧美大片在线观看| 亚洲国产成人高清精品| 久久精品导航| 亚洲一品av免费观看| 欧美激情精品久久久| 黄色工厂这里只有精品| 欧美在线视频免费播放| 亚洲午夜av在线| 欧美系列一区| 亚洲小说欧美另类婷婷| 亚洲靠逼com| 欧美激情自拍| 亚洲乱码国产乱码精品精天堂| 欧美 日韩 国产在线| 久久不见久久见免费视频1| 国产美女扒开尿口久久久| 亚洲综合日韩中文字幕v在线| 日韩视频一区二区三区在线播放免费观看| 久久亚洲综合网| 伊人成人网在线看| 欧美 日韩 国产在线 | 久久精品国产精品亚洲精品| 99re6热只有精品免费观看| 欧美精品一区二区三区在线播放| 亚洲国产日韩欧美在线动漫| 欧美大秀在线观看| 欧美国产一区在线| 亚洲午夜国产一区99re久久| 一本久道久久久| 国产精品一级在线| 久久手机精品视频| 美女91精品| av成人手机在线| 亚洲资源在线观看| 激情视频亚洲| 亚洲精品在线观看免费| 国产精品日韩欧美一区二区| 久久在精品线影院精品国产| 免费日韩精品中文字幕视频在线| 亚洲精品综合| 亚洲一区二区视频在线| 国产真实乱子伦精品视频| 亚洲高清中文字幕| 欧美日韩妖精视频| 久久久亚洲国产美女国产盗摄| 麻豆成人综合网| 亚洲免费视频观看| 久久婷婷蜜乳一本欲蜜臀| 一本色道久久综合亚洲精品高清 | 欧美在线三级| 亚洲精品国产精品国自产在线| 一区二区三区四区精品| 国产亚洲欧美另类中文| 亚洲国产精品t66y| 国产精品美腿一区在线看| 看片网站欧美日韩| 欧美色欧美亚洲高清在线视频| 久久噜噜噜精品国产亚洲综合| 欧美精品在线免费| 久久一区二区三区四区| 国产精品福利久久久| 欧美高清不卡在线| 国产欧美日韩亚洲精品| 亚洲精品一区二区三区不| 国产在线精品成人一区二区三区 | 性做久久久久久免费观看欧美 | 欧美91福利在线观看| 亚洲欧美日韩国产成人| 欧美91视频| 久久综合狠狠综合久久激情| 国产精品av免费在线观看| 亚洲国产精品va在看黑人| 国产亚洲欧美日韩一区二区| 亚洲免费精品| 欧美日韩精品一二三区| 六月天综合网| 国产欧美日韩中文字幕在线| 亚洲免费播放| 亚洲国产色一区| 久久精品99国产精品| 午夜欧美不卡精品aaaaa| 欧美激情第五页| 欧美福利小视频| 影音国产精品| 欧美一级电影久久| 欧美在现视频| 国产精品一区免费在线观看| 亚洲视频免费在线| 在线亚洲激情| 欧美日韩国产三级| 亚洲人成亚洲人成在线观看图片| 亚洲国产成人av在线| 久久久久久9| 欧美成人免费全部观看天天性色| 激情另类综合| 久久久国产精彩视频美女艺术照福利 | 欧美视频不卡| 亚洲另类春色国产| 一本色道久久综合| 欧美午夜宅男影院| 亚洲一区精品在线| 久久精品国产99国产精品| 国产一区二区丝袜高跟鞋图片| 欧美一级久久久| 麻豆亚洲精品| 亚洲久久一区二区| 国产精品成人播放| 欧美在线免费视频| 欧美高清视频一区二区三区在线观看 | 女主播福利一区| 亚洲国产精品成人va在线观看| 亚洲毛片av| 国产精品盗摄久久久| 亚洲欧美一区二区三区久久 | 欧美片在线观看| 一本色道久久综合亚洲精品不卡| 亚洲欧美一区二区精品久久久| 国产日韩精品电影| 久热这里只精品99re8久| 亚洲精品日韩在线观看| 性欧美办公室18xxxxhd| 永久久久久久| 欧美日韩精品在线播放| 午夜精品影院在线观看| 欧美成人免费在线视频| 一区二区高清视频| 国产亚洲在线观看| 欧美激情综合网| 午夜亚洲性色福利视频| 欧美国产国产综合| 午夜精品久久久久99热蜜桃导演| 激情小说另类小说亚洲欧美| 欧美激情四色| 久久精品女人| 一二三四社区欧美黄| 美腿丝袜亚洲色图| 亚洲一区激情| 最新中文字幕亚洲| 国产麻豆日韩欧美久久| 亚洲美女视频在线免费观看| 欧美在线免费观看亚洲| 亚洲精品午夜| 精品av久久707| 国产精品国产自产拍高清av王其| 久久三级视频| 亚洲欧美日韩天堂| 亚洲美女诱惑| 亚洲国产精品久久久久婷婷884 | 蜜臀av一级做a爰片久久 | 亚洲国内在线| 国产色综合久久| 国产精品久久久久免费a∨大胸| 欧美福利在线观看| 久久久久久久久久看片| 亚洲欧美中文日韩在线| 中文精品一区二区三区|