• <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>
            Fork me on GitHub
            隨筆 - 215  文章 - 13  trackbacks - 0
            <2016年8月>
            31123456
            78910111213
            14151617181920
            21222324252627
            28293031123
            45678910


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

            常用鏈接

            留言簿(1)

            隨筆分類

            隨筆檔案

            相冊(cè)

            Awesome

            Blog

            Book

            GitHub

            Link

            搜索

            •  

            積分與排名

            • 積分 - 215445
            • 排名 - 118

            最新評(píng)論

            閱讀排行榜

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

            Server的解耦—通過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ì)說我剛實(shí)習(xí)的時(shí)候就是這么干的。。。)
            所以這一篇主要說說如何設(shè)計(jì)Sever的內(nèi)部邏輯,將Server處理Client發(fā)送信息的這部分邏輯與Sevrer處理Socket連接的邏輯進(jìn)行解耦~
            這一塊的實(shí)現(xiàn)靈感主要是在讀一個(gè)HTTP開源框架: Beego  的源代碼的時(shí)候產(chǎn)生的,Beego的整個(gè)架構(gòu)就是高度解耦的,這里引用一下作者的介紹:
            beego 是基于八大獨(dú)立的模塊構(gòu)建的,是一個(gè)高度解耦的框架。當(dāng)初設(shè)計(jì) beego 的時(shí)候就是考慮功能模塊化,用戶即使不使用 beego 的 HTTP 邏輯,也依舊可以使用這些獨(dú)立模塊,例如:你可以使用 cache 模塊來做你的緩存邏輯;使用日志模塊來記錄你的操作信息;使用 config 模塊來解析你各種格式的文件。所以 beego 不僅可以用于 HTTP 類的應(yīng)用開發(fā),在你的 socket 游戲開發(fā)中也是很有用的模塊,這也是 beego 為什么受歡迎的一個(gè)原因。大家如果玩過樂高的話,應(yīng)該知道很多高級(jí)的東西都是一塊一塊的積木搭建出來的,而設(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)部在通過路由進(jìn)行參數(shù)的過濾,然后路由根據(jù)用戶發(fā)來的參數(shù)判斷調(diào)用哪個(gè)Controller執(zhí)行相關(guān)的邏輯,并在controller里調(diào)用相關(guān)的模塊實(shí)現(xiàn)功能。通過這種方式,Beego成功的將所有模塊都獨(dú)立出來,也就是astaxie所說的“樂高積木化”。
                   在這里,我們可以仿照Beego的架構(gòu),在Server內(nèi)部加入一層Router,通過Router對(duì)通過Socket發(fā)來的信息進(jìn)通過我們?cè)O(shè)定的規(guī)則行的判斷后,調(diào)用相關(guān)的Controller進(jìn)行任務(wù)的分發(fā)處理。在這個(gè)過程中不僅Controller彼此獨(dú)立,匹配規(guī)則和Controller之間也是相互獨(dú)立的。
                   下面給出Router的實(shí)現(xiàn)代碼,其中Msg的結(jié)構(gòu)對(duì)應(yīng)的是Json字符串,當(dāng)然考慮到實(shí)習(xí)公司現(xiàn)在也在用這個(gè),修改了一部分,不過核心思路是一樣的哦:

            復(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") 
                } 

                  通過自定義接口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ā)來信息,我們就可以讓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ā)來的json類型是mirror的時(shí)候,將Client發(fā)來的信息原樣返回:
            復(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ì)良好的日志模塊,不論是開發(fā)Server時(shí)的調(diào)試,還是運(yùn)行時(shí)候的維護(hù),都是非常有幫助的。
            因?yàn)檫@里寫的是一個(gè)比較簡化的Server框架,因此我選擇對(duì)Golang本身的log庫進(jìn)行擴(kuò)充,從而實(shí)現(xiàn)一個(gè)簡單的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的部署。不過在日常開發(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)刷新爬蟲等等,在這里我設(shè)計(jì)了一個(gè)Task接口,通過類似于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 思月行云 閱讀(158) 評(píng)論(0)  編輯 收藏 引用 所屬分類: Golang
            亚洲精品国产自在久久| 欧美亚洲另类久久综合婷婷| 久久精品国产一区二区 | 国产福利电影一区二区三区久久久久成人精品综合 | 国产亚洲美女精品久久久| 青草国产精品久久久久久| 国产69精品久久久久观看软件| 欧美久久天天综合香蕉伊| 精品无码久久久久久国产| 99久久精品费精品国产| 91久久精品无码一区二区毛片| 久久99国产精品一区二区| WWW婷婷AV久久久影片| 国产精品9999久久久久| .精品久久久麻豆国产精品 | 99久久久国产精品免费无卡顿| 777米奇久久最新地址| 久久99精品国产麻豆宅宅| 日韩精品久久久久久| 久久e热在这里只有国产中文精品99| 久久91这里精品国产2020| 免费一级欧美大片久久网| 97视频久久久| 久久婷婷成人综合色综合| 久久国产精品久久国产精品| 久久久国产精品网站| 久久久精品国产亚洲成人满18免费网站| 久久久久国产精品三级网| 亚洲国产成人久久综合野外 | 久久久噜噜噜久久熟女AA片| 国产成年无码久久久久毛片| 色成年激情久久综合| 午夜肉伦伦影院久久精品免费看国产一区二区三区 | 亚洲AV无码久久精品成人| 2020久久精品国产免费| 久久国产一片免费观看| 国色天香久久久久久久小说| 亚洲国产成人久久综合碰碰动漫3d | 久久亚洲日韩精品一区二区三区 | 伊人热热久久原色播放www| 日韩精品久久久久久久电影蜜臀|