GO實現Redis:GO實現TCP伺服器(1)

来源:https://www.cnblogs.com/csgopher/archive/2023/03/23/17248642.html
-Advertisement-
Play Games

本文實現一個Echo TCP Server 完整代碼:https://github.com/csgopher/go-redis interface/tcp/Handler.go type Handler interface { Handle(ctx context.Context, conn net ...


interface/tcp/Handler.go

type Handler interface {
   Handle(ctx context.Context, conn net.Conn)
   Close() error
}
  • Handler:業務邏輯的處理介面
    • Handle(ctx context.Context, conn net.Conn) 處理連接

tcp/server.go

type Config struct {
    Address string
}

func ListenAndServeWithSignal(cfg *Config, handler tcp.Handler) error {
    closeChan := make(chan struct{})
    listen, err := net.Listen("tcp", cfg.Address)
    if err != nil {
       return err
   }
    logger.Info("start listen")
    ListenAndServe(listen, handler, closeChan)
    return nil
}

func ListenAndServe(listener net.Listener,
                    handler tcp.Handler,
                    closeChan <-chan struct{}) {
    ctx := context.Background()
    var waitDone sync.WaitGroup
    for true {
        conn, err := listener.Accept()
        if err != nil {
            break
        }
        logger.Info("accept link")
        waitDone.Add(1)
        go func() {
            defer func() {
                waitDone.Done()
            }()
            handler.Handler(ctx, conn)
        }()
    }
    waitDone.Wait()
}
  • Config:啟動tcp伺服器的配置
    • Address:監聽地址
  • ListenAndServe:ctx是上下文,可以傳遞一些參數。死迴圈中接收到新連接時,讓一個協程去處理連接
  • 如果listener.Accept()出錯了就會break跳出來,這時候需要等待已經服務的客戶端退出。使用WaitGroup等待客服端退出

func ListenAndServe(listener net.Listener,
                    handler tcp.Handler,
                    closeChan <-chan struct{}) {

    go func() {
       <-closeChan
       logger.Info("shutting down...")
       _ = listener.Close()
       _ = handler.Close()
   }()

    defer func() {
       _ = listener.Close()
       _ = handler.Close()
   }()

    ......
}

listener和handler在退出的時候需要關掉。如果用戶直接kill掉了程式,我們也需要關掉listener和handler,這時候要使用closeChan,一旦接收到關閉信號,就執行關閉邏輯

func ListenAndServeWithSignal(cfg *Config, handler tcp.Handler) error {

    closeChan := make(chan struct{})
    sigCh := make(chan os.Signal)
    signal.Notify(sigCh, syscall.SIGHUP, syscall.SIGQUIT, syscall.SIGTERM, syscall.SIGINT)
    go func() {
       sig := <-sigCh
       switch sig {
          case syscall.SIGHUP, syscall.SIGQUIT, syscall.SIGTERM, syscall.SIGINT:
          closeChan <- struct{}{}
      }
   }()
    listen, err := net.Listen("tcp", cfg.Address)
    if err != nil {
       return err
   }
    logger.Info("start listen")
    ListenAndServe(listen, handler, closeChan)
    return nil
}

當系統對程式發送信號時,sigCh會接收到信號

tcp/echo.go

type EchoHandler struct {
   activeConn sync.Map
   closing    atomic.Boolean
}

EchoHandler:

  • activeConn:記錄連接
  • closing:是否正在關閉,有併發競爭,使用atomic.Boolean

type EchoClient struct {
   Conn    net.Conn
   Waiting wait.Wait
}

func (c *EchoClient) Close() error {
	c.Waiting.WaitWithTimeout(10 * time.Second)
	_ = c.Conn.Close()
	return nil
}

EchoClient:一個客戶端就是一個連接。Close方法關閉客戶端連接,超時時間設置為10s

func MakeHandler() *EchoHandler {
	return &EchoHandler{}
}

func (h *EchoHandler) Handle(ctx context.Context, conn net.Conn) {
   // 連接正在關閉,不接收新連接
   if h.closing.Get() {
      _ = conn.Close()
   }

   client := &EchoClient{
      Conn: conn,
   }
   h.activeConn.Store(client, struct{}{})

   reader := bufio.NewReader(conn)
   for {
      msg, err := reader.ReadString('\n')
      if err != nil {
         if err == io.EOF {
            logger.Info("connection close")
            h.activeConn.Delete(client)
         } else {
            logger.Warn(err)
         }
         return
      }
      // 正在處理業務,不要關掉
      client.Waiting.Add(1)
      // 將數據原封不動寫回去,測試
      b := []byte(msg)
      _, _ = conn.Write(b)
      client.Waiting.Done()
   }
}

func (h *EchoHandler) Close() error {
   logger.Info("handler shutting down...")
   h.closing.Set(true)
   h.activeConn.Range(func(key interface{}, val interface{}) bool {
      client := key.(*EchoClient)
      _ = client.Close()
      return true
   })
   return nil
}
  • MakeEchoHandler:創建EchoHandler
  • Handle:處理客戶端的連接。
    • 1.連接正在關閉時,不接收新連接
    • 2.存儲新連接,value用空結構體
    • 3.使用緩存區接收用戶發來的數據,使用\n作為結束的標誌
  • Close:將所有客戶端連接關掉

main.go

const configFile string = "redis.conf"

var defaultProperties = &config.ServerProperties{
   Bind: "0.0.0.0",
   Port: 6379,
}

func fileExists(filename string) bool {
   info, err := os.Stat(filename)
   return err == nil && !info.IsDir()
}

func main() {
   logger.Setup(&logger.Settings{
      Path:       "logs",
      Name:       "godis",
      Ext:        "log",
      TimeFormat: "2022-02-02",
   })

   if fileExists(configFile) {
      config.SetupConfig(configFile)
   } else {
      config.Properties = defaultProperties
   }

   err := tcp.ListenAndServeWithSignal(
      &tcp.Config{
         Address: fmt.Sprintf("%s:%d",
            config.Properties.Bind,
            config.Properties.Port),
      },
      EchoHandler.MakeHandler())
   if err != nil {
      logger.Error(err)
   }
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • SpringBoot內置Tomcat的配置和切換 1.基本介紹 SpringBoot支持的webServer:Tomcat,Jetty,Undertow 因為在spring-boot-starter-web中,預設導入的是tomcat,因此啟動時使用的web容器就是tomcat。 同時 Spring ...
  • 最近在看了《微信背後的產品觀 - 張小龍手抄版》,其中有段話如下: 用戶需求是零散的,解決方案是歸納抽象的過程 那如何歸納抽象呢?是否有一定的實踐方法論呢?經過一輪探討和學習,有這些答案: 5 Whys 分析法 U 型思考法 等等 二、5 Whys 分析法 5 Whys 法,最初由豐田佐吉開發,併在 ...
  • 多線程 多線程概述 多線程就是電腦用時運行多個任務 但實質上,同一個時間點,只會運行一個任務,只是電腦在不同任務之間來回切換而已。 併發和並行 並行:在同一時間,多個任務分別在多個CPU上進行。 併發:在同一時間,多個任務在同一個CPU交替進行。 線程和進程 進程 獨立性:進程是一個獨立運行的應 ...
  • SpringBoot中註入Servlet&Filter&Listener 1.基本介紹 文檔:SpringBoot中註入Servlet&Filter&Listener 考慮到實際開發業務非常複雜和相容問題,SpringBoot支持將Servlet、Filter、Listener註入spring容器中 ...
  • Styled Components 備忘清單 IT寶庫整理的Styled Components快速參考備忘單提供了使用 CSS in JS 工具的各種方法入門,為開發人員分享快速參考備忘單。 開發速查表大綱 入門 安裝 快速開始 根據 Props 適配 擴展樣式 擴展樣式改變標簽 (as) 自定義組 ...
  • Stylus 備忘清單 IT寶庫整理的Stylus開發速查備忘單旨在快速理解 stylus 所涉及的主要概念,顯示了它的常用方法使用清單入門,為開發人員分享快速參考備忘單。 開發速查表大綱 入門 介紹 支持 CSS 嵌套語法 支持類 python 縮進語法 混合 Mixins 變數 Variable ...
  • 最近工作中經常使用Teamcenter、NX集成開發的情況,因此在這裡記錄UF_UGMGR函數的使用。使用UF_UGMGR相關函數需要有Teamcenter使用經驗,理解Teamcenter中文件夾、偽文件夾、零組件、零組件版本、數據集、關係、表單、命名引用等對象。 相關的可以看幫助中 Teamce ...
  • 本篇文章將深入介紹 Yarn 三種調度器。Yarn 本身作為資源管理和調度服務,其中的資源調度模塊更是重中之重。下麵將介紹 Yarn 中實現的調度器功能,以及內部執行邏輯。 ...
一周排行
    -Advertisement-
    Play Games
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...
  • 目錄前言PostgreSql安裝測試額外Nuget安裝Person.cs模擬運行Navicate連postgresql解決方案Garnet為什麼要選擇Garnet而不是RedisRedis不再開源Windows版的Redis是由微軟維護的Windows Redis版本老舊,後續可能不再更新Garne ...
  • C#TMS系統代碼-聯表報表學習 領導被裁了之後很快就有人上任了,幾乎是無縫銜接,很難讓我不想到這早就決定好了。我的職責沒有任何變化。感受下來這個系統封裝程度很高,我只要會調用方法就行。這個系統交付之後不會有太多問題,更多應該是做小需求,有大的開發任務應該也是第二期的事,嗯?怎麼感覺我變成運維了?而 ...
  • 我在隨筆《EAV模型(實體-屬性-值)的設計和低代碼的處理方案(1)》中介紹了一些基本的EAV模型設計知識和基於Winform場景下低代碼(或者說無代碼)的一些實現思路,在本篇隨筆中,我們來分析一下這種針對通用業務,且只需定義就能構建業務模塊存儲和界面的解決方案,其中的數據查詢處理的操作。 ...
  • 對某個遠程伺服器啟用和設置NTP服務(Windows系統) 打開註冊表 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\W32Time\TimeProviders\NtpServer 將 Enabled 的值設置為 1,這將啟用NTP伺服器功 ...
  • title: Django信號與擴展:深入理解與實踐 date: 2024/5/15 22:40:52 updated: 2024/5/15 22:40:52 categories: 後端開發 tags: Django 信號 松耦合 觀察者 擴展 安全 性能 第一部分:Django信號基礎 Djan ...
  • 使用xadmin2遇到的問題&解決 環境配置: 使用的模塊版本: 關聯的包 Django 3.2.15 mysqlclient 2.2.4 xadmin 2.0.1 django-crispy-forms >= 1.6.0 django-import-export >= 0.5.1 django-r ...
  • 今天我打算整點兒不一樣的內容,通過之前學習的TransformerMap和LazyMap鏈,想搞點不一樣的,所以我關註了另外一條鏈DefaultedMap鏈,主要調用鏈為: 調用鏈詳細描述: ObjectInputStream.readObject() DefaultedMap.readObject ...
  • 後端應用級開發者該如何擁抱 AI GC?就是在這樣的一個大的浪潮下,我們的傳統的應用級開發者。我們該如何選擇職業或者是如何去快速轉型,跟上這樣的一個行業的一個浪潮? 0 AI金字塔模型 越往上它的整個難度就是職業機會也好,或者說是整個的這個運作也好,它的難度會越大,然後越往下機會就會越多,所以這是一 ...
  • @Autowired是Spring框架提供的註解,@Resource是Java EE 5規範提供的註解。 @Autowired預設按照類型自動裝配,而@Resource預設按照名稱自動裝配。 @Autowired支持@Qualifier註解來指定裝配哪一個具有相同類型的bean,而@Resourc... ...