由于从之前的章节,您已经知道了,TCP 代表传输控制协议,并且它的主要特点是可靠性。每个包的 TCP 头部包含源端口和目标端口字段。这俩个字段,加上源和目标 IP 地址,组合成唯一标识单个 TCP 连接。这节实现的 TCP 客户端命名为 TCPclient.go,它由以下四部分构成。第一部分代码:
package mainimport ("bufio""fmt""net""os""strings")
第二部分代码:
func main(){arguments := os.Argsif len(arguments) == 1 {fmt.Println("Please provide host:port.")return}CONNECT := arguments[1]c, err := net.Dial("tcp", CONNECT)if err != nil {fmt.Println(err)return}
net.Dial() 方法连接远端服务器。这个方法的第一个参数定义使用的网络连接类型,第二个参数定义服务器地址,并且必须包含端口号。第一个参数的有效值为 tcp,tcp4 (IPv4-only),tcp6 (IPv6-ony),udp, udp4 (IPv4-only),udp6 (IPv6-only),ip,ip4 (IPv4-only),ip6 (IPv6-only),Unix (Unix sockets),Unixgram 和 Unixpacket。
第三部分代码:
for {reader := bufio.NewReader(os.Stdin)fmt.Print(">> ")text, _ := reader.ReadString('\n')fmt.Fprintf(c, text + "\n")
前面这段代码用于从用户获取输入,使用 os.Stdin
文件进行验证读取。忽略从 reader.ReadString()
返回的 error
值不是一个好的实现,但它节省了空间。当然,你永远不要在正式软件中这样做。
最后一段代码:
message, _ := bufio.NewReader(c).ReadString('\n')fmt.Print("->: " + message)if strings.TrimSpace(string(text)) == "STOP" {fmt.Println("TCP client exiting...")return}}}
为了测试,TCPclient.go
将连接由 netcat(1)
实现的TCP 服务器,它将产生以下输出:
$ go run TCPclient.go 8001dial tcp: address 8001: missing port in address$ go run TCPclient.go localhost:8001>> Hello from TCPclient.go!->: Hi from nc!>> STOP->:TCP client exiting...
请注意,给定协议如 TCP 和 UDP 的客户端本质上是可以通用的,这意味着它能够与支持其协议的多种服务器通信。您很快会看到,并不是使用指定协议的服务端应用必须实现指定的功能