答案:使用Go的net包实现端口扫描,通过DialTimeout检测连接,结合goroutine并发扫描多端口。示例代码包含scanPort检测单个端口,portScan并发扫描并输出开放端口,完整程序可扫描scanme.nmap.org的指定端口并统计耗时。

实现一个基本的端口扫描器在Golang中非常直接,主要利用
net包提供的网络功能。你可以通过尝试与目标IP的特定端口建立TCP连接,判断端口是否开放。下面是一个简单但实用的端口扫描器实现步骤和代码示例。
1. 理解端口扫描的基本原理
端口扫描的核心是向目标主机的某个端口发起连接请求。如果连接成功(无错误返回),说明该端口处于开放状态;如果连接失败或超时,则认为端口关闭或被防火墙拦截。
在Go中,使用
net.DialTimeout()可以设置连接超时,避免程序卡住。
2. 编写基础端口扫描函数
以下是一个扫描单个端口的函数示例:
立即学习“go语言免费学习笔记(深入)”;
func scanPort(protocol, host string, port int, timeout time.Duration) bool {
address := fmt.Sprintf("%s:%d", host, port)
conn, err := net.DialTimeout(protocol, address, timeout)
if err != nil {
return false
}
conn.Close()
return true
}
这个函数尝试连接指定的主机和端口,如果成功则返回
true,否则返回
false。
3. 并发扫描多个端口
为了提高效率,可以使用Go的goroutine并发扫描多个端口。通过
sync.WaitGroup控制并发流程,使用
channel收集结果。
func portScan(host string, ports []int, timeout time.Duration) {
var wg sync.WaitGroup
result := make(chan int)
// 启动结果收集goroutine
go func() {
for openPort := range result {
fmt.Printf("Port %d is open\n", openPort)
}
}()
// 并发扫描每个端口
for _, port := range ports {
wg.Add(1)
go func(p int) {
defer wg.Done()
if scanPort("tcp", host, p, timeout) {
result <- p
}
}(port)
}
wg.Wait()
close(result)}
4. 完整可运行示例
将以上部分组合成一个完整的程序:
package mainimport ( "fmt" "net" "sync" "time" )
func scanPort(protocol, host string, port int, timeout time.Duration) bool { address := fmt.Sprintf("%s:%d", host, port) conn, err := net.DialTimeout(protocol, address, timeout) if err != nil { return false } conn.Close() return true }
func portScan(host string, ports []int, timeout time.Duration) { var wg sync.WaitGroup result := make(chan int)
go func() { for openPort := range result { fmt.Printf("Port %d is open\n", openPort) } }() for _, port := range ports { wg.Add(1) go func(p int) { defer wg.Done() if scanPort("tcp", host, p, timeout) { result <- p } }(port) } wg.Wait() close(result)}
func main() { target := "scanme.nmap.org" // 可公开测试的目标 portList := []int{22, 80, 443, 8080}
fmt.Printf("Scanning %s...\n", target) start := time.Now() portScan(target, portList, 2*time.Second) fmt.Printf("Scan completed in %v\n", time.Since(start))}
这段代码会并发扫描指定的几个端口,并输出开放的端口。你也可以将
portList扩展为1-1024的范围来扫描常用端口。
基本上就这些。这个扫描器虽然简单,但已经具备基本功能。你可以在此基础上添加功能,比如支持端口范围输入、扫描协议选择、结果保存到文件等。注意:扫描非自己控制的主机可能违反法律或服务条款,请确保在授权范围内使用。










