亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

golang高并發(fā)限流操作 ping / telnet

 更新時間:2020年12月21日 10:41:36   作者:出逃的迪斯尼貓咪  
這篇文章主要介紹了golang高并發(fā)限流操作 ping / telnet,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧

需求

當需要同時ping/telnet多個ip時,可以通過引入ping包/telnet包實現(xiàn),也可以通過go調用cmd命令實現(xiàn),不過后者調用效率較差,所以這里選擇ping包和telnet包

還有就是高并發(fā)的問題,可以通過shell腳本或者go實現(xiàn)高并發(fā),所以我選擇的用go自帶的協(xié)程實現(xiàn),但是如果要同時處理1000+個ip,考慮到機器的性能,需要ratelimit控制開辟的go協(xié)程數(shù)量,這里主要寫一下我的建議和淌過的坑

ping

參考鏈接: https://github.com/sparrc/go-ping

import "github.com/sparrc/go-ping"
import "time"
func (p *Ping) doPing(timeout time.Duration, count int, ip string) (err error) {
 pinger, cmdErr := ping.NewPinger(ip)
 if cmdErr != nil {
  glog.Error("Failed to ping " + p.ipAddr)
  err = cmdErr
  return
 }
 pinger.Count = count
 pinger.Interval = time.Second
 pinger.Timeout = timeout
 // true的話,代表是標準的icmp包,false代表可以丟包類似udp
 pinger.SetPrivileged(false)
 // 執(zhí)行
 pinger.Run() 
 // 獲取ping后的返回信息
 stats := pinger.Statistics() 
 //延遲
 latency = float64(stats.AvgRtt) 
 // 標準的往返總時間 
 jitter = float64(stats.StdDevRtt) 
 //丟包率 
 packetLoss = stats.PacketLoss 
 return
}

注意: pinger.Run() 這里執(zhí)行的時候是阻塞的,如果并發(fā)量大的時候,程序會卡死在這里,所以當有高并發(fā)的需求時建議如下處理:

go pinger.Run()

time.Sleep(timeout)

telnet

package main
import (
 "github.com/reiver/go-telnet"
)
func doTelnet(ip string, port int) {
 var caller telnet.Caller = telnet.StandardCaller
 address := ip + ":"+ strconv.Itoa(port)
 // DialToAndCall 檢查連通性并且調用
 telnet.DialToAndCall(address, caller)
 }
}

bug出現(xiàn)報錯:

lookup tcp/: nodename nor servname provided, or not known

解決:

修改string(port)為strconv.Itoa(port)

DialToAndCall這種方式telnet無法設置超時時間,默認的超時時間有1分鐘,所以使用DialTimeout這個方式實現(xiàn)telnet

import "net"
func doTelnet(ip string, ports []string) map[string]string {
 // 檢查 emqx 1883, 8083, 8080, 18083 端口
 results := make(map[string]string)
 for _, port := range ports {
 address := net.JoinHostPort(ip, port)
 // 3 秒超時
 conn, err := net.DialTimeout("tcp", address, 3*time.Second)
 if err != nil {
 results[port] = "failed"
 } else {
 if conn != nil {
 results[port] = "success"
 _ = conn.Close()
 } else {
 results[port] = "failed"
 }
 }
 }
 return results
}

shell高并發(fā)

本質就是讀取ip.txt文件里的ip,然后調用ping方法,實現(xiàn)高并發(fā)也是借助&遍歷所有的ip然后同一交給操作系統(tǒng)去處理高并發(fā)

while read ip
do
 {
 doPing(ip)
 } &
done < ip.txt

go高并發(fā)限速

import (
 "context"
 "fmt"
 "log"
 "time"
 "sync"
 "golang.org/x/time/rate"
)
func Limit(ips []string)([]string, []string, error) {
 //第一個參數(shù)是每秒鐘最大的并發(fā)數(shù),第二個參數(shù)是桶的容量,第一次的時候每秒可執(zhí)行的數(shù)量就是桶的容量,建議這兩個值都寫成一樣的
 r := rate.NewLimiter(10, 10)
 ctx := context.Background()
 
 wg := sync.WaitGroup{}
 wg.Add(len(ips))
 
 lock := sync.Mutex{}
 var success []string
 var fail []string
 
 defer wg.Done()
 for _,ip:=range ips{
 //每次消耗2個,放入一個,消耗完了還會放進去,如果初始是5個,所以這段代碼再執(zhí)行到第4次的時候筒里面就空了,如果當前不夠取兩個了,本次就不取,再放一個進去,然后返回false
 err := r.WaitN(ctx, 2)
 if err != nil {
  log.Fatal(err)
 }
 go func(ip string) {
 defer func() {
 wg.Done()
 }()
 err := doPing(time.Second, 2, ip)
 lock.Lock()
 defer lock.Unlock()
 if err != nil {
 fail = append(fail, ip)
 return
 } else {
 success = append(success, ip)
 }
 }(ip)
 }
 // wait等待所有go協(xié)程結束
 wg.wait()
 return success,fail,nil
}
func main() {
 ips := [2]string{"192.168.1.1","192.168.1.2"}
 success,fail,err := Limit(ips)
 if err != nil {
 fmt.Printf("ping error")
 }
}

這里注意一個并發(fā)實現(xiàn)的坑,在for循環(huán)里使用goroutine時要把遍歷的參數(shù)傳進去才能保證每個遍歷的參數(shù)都被執(zhí)行,否則只能執(zhí)行一次

(拓展)管道、死鎖

先看個例子:

func main() {
 go print() // 啟動一個goroutine
 print()
}
func print() {
fmt.Println("*******************")
}

輸出結果:

*******************

沒錯,只有一行,因為當go開辟一個協(xié)程想去執(zhí)行print方法時,主函數(shù)已經(jīng)執(zhí)行完print并打印出來,所以goroutine還沒有機會執(zhí)行程序就已經(jīng)結束了,解決這個問題可是在主函數(shù)里加time.sleep讓主函數(shù)等待goroutine執(zhí)行完,也可以使用WaitGroup.wait等待goroutine執(zhí)行完,還有一種就是信道

信道分無緩沖信道和緩沖信道

無緩沖信道

無緩沖信道也就是定義長度為0的信道,存入一個數(shù)據(jù),從無緩沖信道取數(shù)據(jù),若信道中無數(shù)據(jù),就會阻塞,還可能引發(fā)死鎖,同樣數(shù)據(jù)進入無緩沖信道, 如果沒有其他goroutine來拿走這個數(shù)據(jù),也會阻塞,記住無緩沖數(shù)據(jù)并不存儲數(shù)據(jù)

func main() {
 var channel chan string = make(chan string)
 go func(message string) {
 channel<- message // 存消息
 }("Ping!")
 fmt.Println(<-messages) // 取消息
}

緩存信道

顧名思義,緩存信道可以存儲數(shù)據(jù),goroutine之間不會發(fā)生阻塞,for循環(huán)讀取信道中的數(shù)據(jù)時,一定要判斷當管道中不存在數(shù)據(jù)時的情況,否則會發(fā)生死鎖,看個例子

channel := make(chan int, 3)
channel <- 1
channel <- 2
channel <- 3
// 顯式關閉信道
close(channel)
for v := range channel {
 fmt.Println(v)
 // 如果現(xiàn)有數(shù)據(jù)量為0,跳出循環(huán),與顯式關閉隧道效果一樣,選一個即可
 if len(ch) <= 0 { 
 break
 }
}

但是這里有個問題,信道中數(shù)據(jù)是可以隨時存入的,所以我們遍歷的時候無法確定目前的個數(shù)就是信道的總個數(shù),所以推薦使用select監(jiān)聽信道

// 創(chuàng)建一個計時信道
timeout := time.After(1 * time.Second) 
// 監(jiān)聽3個信道的數(shù)據(jù)
select {
 case v1 := <- c1: fmt.Printf("received %d from c1", v1)
 case v2 := <- c2: fmt.Printf("received %d from c2", v2)
 case v3 := <- c3: fmt.Printf("received %d from c3", v3)
 case <- timeout: 
 is_timeout = true // 超時
 break
 }
}

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。

相關文章

  • golang等待觸發(fā)事件的實例

    golang等待觸發(fā)事件的實例

    這篇文章主要介紹了golang等待觸發(fā)事件的實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • Go語言拼接URL路徑的三種方法

    Go語言拼接URL路徑的三種方法

    本文主要介紹了Go語言拼接URL路徑的三種方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-03-03
  • Go?defer?去掉閉包函數(shù)及用法分析

    Go?defer?去掉閉包函數(shù)及用法分析

    這篇文章主要為大家介紹了Go?defer?去掉閉包函數(shù)及用法分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-07-07
  • Go語言nil標識符(空值/零值)

    Go語言nil標識符(空值/零值)

    本文主要介紹了Go語言nil標識符(空值/零值),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-03-03
  • GO語言開發(fā)終端命令行小工具改進更新

    GO語言開發(fā)終端命令行小工具改進更新

    這篇文章主要為大家介紹了GO語言開發(fā)終端命令行小工具的改進更新,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2024-01-01
  • go實現(xiàn)一個分布式限流器的方法步驟

    go實現(xiàn)一個分布式限流器的方法步驟

    項目中需要對api的接口進行限流,本文主要介紹了go實現(xiàn)一個分布式限流器的方法步驟,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-01-01
  • Golang unsafe.Sizeof函數(shù)代碼示例使用解析

    Golang unsafe.Sizeof函數(shù)代碼示例使用解析

    這篇文章主要為大家介紹了Golang unsafe.Sizeof函數(shù)代碼示例使用解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-12-12
  • golang?gorm實現(xiàn)get請求查詢案例測試

    golang?gorm實現(xiàn)get請求查詢案例測試

    這篇文章主要為大家介紹了golang?gorm實現(xiàn)get請求查詢案例測試,
    2022-04-04
  • Go語言使用對稱加密的示例詳解

    Go語言使用對稱加密的示例詳解

    在項目開發(fā)中,我們經(jīng)常會遇到需要使用對稱密鑰加密的場景,比如客戶端調用接口時,參數(shù)包含手機號、身份證號或銀行卡號等。本文將詳細講解Go語言使用對稱加密的方法,需要的可以參考一下
    2022-06-06
  • Go語言實現(xiàn)配置熱加載的方法分享

    Go語言實現(xiàn)配置熱加載的方法分享

    web項目,經(jīng)常需要熱啟動各種各樣的配置信息,一旦這些服務發(fā)生變更,我們需要重新啟動web server,以使配置生效,實現(xiàn)配置熱加載,本文為大家整理了幾個方法實現(xiàn)這個需求,需要的可以參考下
    2023-05-05

最新評論