go 實現簡易端口掃描的示例
我在代碼裡定義瞭兩個通道,分別用於生產端口和限制連接數,如果不限制連接數,容易被對方檢測到或導致對方服務器不能正常運行。
// 生產端口 var port = make(chan int, 10) // 限制並發數 var connect = make(chan string, 5)
可以使用net庫的Dial函數做為socket客戶端,需要註意的是要設置超時時間,因為若主機不存在,或目標端口是關閉的,往往需要花費數秒才返回錯誤,這樣掃描大量端口時效率會極其低下。在go中可以使用net.Dialer結構體設置超時時間,然後在調用Dial方法:
d := net.Dialer{Timeout: time.Second} dial, err := d.Dial("tcp", target)
隻要err不等於nil,表示目標端口是對外開放的。
完整代碼如下:
package main import ( "fmt" "net" "sync" "time" ) var wg sync.WaitGroup func main() { var start, end int var address string fmt.Printf("請輸入目標IP:> ") fmt.Scan(&address) fmt.Printf("請輸入起始端口:> ") fmt.Scan(&start) fmt.Printf("請輸入結束端口:> ") fmt.Scan(&end) wg.Add(end - start + 1) Run(address, start, end) wg.Wait() fmt.Println("執行完畢") } // 生產端口 var port = make(chan int, 10) // 限制並發數 var connect = make(chan string, 5) func Run(address string, start, end int) { go func() { for i := start; i <= end; i++ { port <- i } }() go func() { // 消費端口 for p := range port { // 往通道寫入目標地址,超過限制並發數會阻塞 connect <- fmt.Sprintf("%s:%d", address, p) } }() go Connect() } func Connect() { // 並發請求 for target := range connect { // 設置超時時間 d := net.Dialer{Timeout: time.Second} dial, err := d.Dial("tcp", target) if err == nil { fmt.Printf("%s 連接成功\n", target) dial.Close() }else{ fmt.Printf("%s 連接失敗\n", target) } wg.Done() } }
這裡端口生產通道不是必須的,隻是為瞭演示消費生產並發模型,當然這還是最簡單的。
在Run函數裡我沒有關閉這兩個通道,按官方的說法是gc會回收不使用的通道,如果要手動關閉,可以定義defer閉包進行close。
我本地運行結果如下:
請輸入目標IP:> 127.0.0.1 請輸入起始端口:> 8080 請輸入結束端口:> 8094 127.0.0.1:8080 連接成功 127.0.0.1:8081 連接成功 127.0.0.1:8082 連接成功 127.0.0.1:8083 連接成功 127.0.0.1:8084 連接成功 127.0.0.1:8085 連接成功 127.0.0.1:8086 連接成功 127.0.0.1:8087 連接成功 127.0.0.1:8088 連接成功 127.0.0.1:8089 連接成功 127.0.0.1:8090 連接成功 127.0.0.1:8091 連接成功 127.0.0.1:8092 連接成功 127.0.0.1:8093 連接失敗 127.0.0.1:8094 連接失敗 執行完畢
這個版本比較簡陋,TCP連接過程也可以簡化,後續再寫另外一篇文章。因為最近在學rust語言,語法內容比較多,所以後面暫時發佈編程相關的文章,提升一下語法熟練度。
以上就是go 實現簡易端口掃描的示例的詳細內容,更多關於go 實現端口掃描的資料請關註WalkonNet其它相關文章!
推薦閱讀:
- Go語言如何輕松編寫高效可靠的並發程序
- golang 並發編程之生產者消費者詳解
- golang 實現並發求和
- golang高並發限流操作 ping / telnet
- 解決golang sync.Wait()不執行的問題