婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > GOLANG使用Context管理關聯goroutine的方法

GOLANG使用Context管理關聯goroutine的方法

熱門標簽:廣州呼叫中心外呼系統 南通如皋申請開通400電話 地圖標注的汽車標 浙江高速公路地圖標注 西部云谷一期地圖標注 江西轉化率高的羿智云外呼系統 學海導航地圖標注 中國地圖標注省會高清 高德地圖標注口訣

一般一個業務很少不用到goroutine的,因為很多方法是需要等待的,例如http.Server.ListenAndServe這個就是等待的,除非關閉了Server或Listener,否則是不會返回的。除非是一個API服務器,否則肯定需要另外起goroutine發起其他的服務,而且對于API服務器來說,在http.Handler的處理函數中一般也需要起goroutine,如何管理這些goroutine,在GOLANG1.7提供context.Context

先看一個簡單的,如果啟動兩個goroutine,一個是HTTP,還有個信號處理的收到退出信號做清理:

wg := sync.WaitGroup{}
defer wg.Wait()

wg.Add(1)
go func() {
  defer wg.Done()

  ss := make(os.Signal, 0)
  signal.Notify(ss, syscall.SIGINT, syscall.SIGTERM)
  for s := ss {
    fmt.Println("Got signal", s)
    break
  }
}()

wg.Add(1)
go func() {
  defer wg.Done()

  svr := http.Server{ Addr:":8080", Handler:nil, }
  fmt.Println(svr.ListenAndServe())
}

很清楚,起了兩個goroutine,然后用WaitGroup等待它們退出。如果它們之間沒有交互,不互相影響,那真的是蠻簡單的,可惜這樣是不行的,因為信號的goroutine收到退出信號后,應該通知server退出。暴力一點的是直接調用svr.Close(),但是如果有些請求還需要取消怎么辦呢?最好用Context了:

wg := sync.WaitGroup{}
defer wg.Wait()

ctx,cancel := context.WithCancel(context.Background())

wg.Add(1)
go func() {
  defer wg.Done()

  ss := make(chan os.Signal, 0)
  signal.Notify(ss, syscall.SIGINT, syscall.SIGTERM)
  select {
  case - ctx.Done():
    return
  case s := - ss:
    fmt.Println("Got signal", s)
    cancel() // 取消請求,通知用到ctx的所有goroutine
    return
  }
}()

wg.Add(1)
go func() {
  defer wg.Done()
  defer cancel()

  svr := http.Server{ Addr:":8080", Handler:nil, }

  go func(){
    select {
    case - ctx.Done():
      svr.Close()
    }
  }

  fmt.Println(svr.ListenAndServe())
}

這個方式可以在新開goroutine時繼續使用,譬如新加一個goroutine,里面讀寫了UDPConn:

wg.Add(1)
go func() {
  defer wg.Done()
  defer cancel()

  var conn *net.UDPConn
  if conn,err = net.Dial("udp", "127.0.0.1:1935"); err != nil {
    fmt.Println("Dial UDP server failed, err is", err)
    return
  }

  fmt.Println(UDPRead(ctx, conn))
}()

UDPRead = func(ctx context.Context, conn *net.UDPConn) (err error) {
  wg := sync.WaitGroup{}
  defer wg.Wait()

  ctx, cancel := context.WithCancel(ctx)

  wg.Add(1)
  go func() {
    defer wg.Done()
    defer cancel()

    for {
      b := make([]byte, core.MTUSize)
      size, _, err := conn.ReadFromUDP(b)
      // 處理UDP包 b[:size]
    }
  }()

  select {
  case -ctx.Done():
    conn.Close()
  }
  return
}

如果只是用到HTTP Server,可以這么寫:

func run(ctx contex.Context) {
  server := http.Server{Addr: addr, Handler: nil}
  go func() {
    select {
    case -ctx.Done():
      server.Close()
    }
  }()

  http.HandleFunc("/api", func(w http.ResponseWriter, r *http.Request) {
  })

  fmt.Println(server.ListenAndServe())
}

如果需要提供一個API來讓服務器退出,可以這么寫:

func run(ctx contex.Context) {
  server := http.Server{Addr: addr, Handler: nil}

  ctx, cancel := context.WithCancel(ctx)
  http.HandleFunc("/quit", func(w http.ResponseWriter, r *http.Request) {
    cancel() // 使用局部的ctx和cancel
  })

  go func() {
    select {
    case -ctx.Done():
      server.Close()
    }
  }()

  fmt.Println(server.ListenAndServe())
}

使用局部的ctx和cancel,可以避免cancel傳入的ctx,只是影響當前的ctx。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • golang中for循環遍歷channel時需要注意的問題詳解
  • golang實現基于channel的通用連接池詳解
  • Golang優雅關閉channel的方法示例
  • golang中單向channel的語法介紹
  • 解決Golang中goroutine執行速度的問題
  • golang gin 框架 異步同步 goroutine 并發操作
  • 關于golang利用channel和goroutine完成統計素數的思路

標簽:吐魯番 德宏 曲靖 常州 貴州 許昌 保定 東營

巨人網絡通訊聲明:本文標題《GOLANG使用Context管理關聯goroutine的方法》,本文關鍵詞  GOLANG,使用,Context,管理,關聯,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《GOLANG使用Context管理關聯goroutine的方法》相關的同類信息!
  • 本頁收集關于GOLANG使用Context管理關聯goroutine的方法的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 枝江市| 张家口市| 元谋县| 邓州市| 屏东市| 南丹县| 饶河县| 大兴区| 桐柏县| 金门县| 遵义市| 章丘市| 榆树市| 萝北县| 秦皇岛市| 陆川县| 新干县| 石楼县| 赤壁市| 宜丰县| 延庆县| 宁河县| 城口县| 都兰县| 连城县| 保靖县| 喀什市| 福清市| 乐至县| 昌黎县| 安顺市| 西华县| 奉化市| 故城县| 开鲁县| 道真| 吴忠市| 丹东市| 贡觉县| 沿河| 元江|