Golang网络编程-套接字(socket)篇
版权声明:原创作品,谢绝转载!否则将追究法律责任。
一.网络概述
1>.什么是协议
从应用的角度出发,协议可理解为“规则”,是数据传输和数据的解释的规则。假设,A、B双方欲传输文件。规定:
第一次,传输文件名,接收方接收到文件名,应答OK给传输方;
第二次,发送文件的尺寸,接收方接收到该数据再次应答一个OK;
第三次,传输文件内容。同样,接收方接收数据完成后应答OK表示文件内容接收成功。
由此,无论A、B之间传递何种文件,都是通过三次数据传输来完成。A、B之间形成了一个最简单的数据传输规则。双方都按此规则发送、接收数据。A、B之间达成的这个相互遵守的规则即为协议。
这种仅在A、B之间被遵守的协议称之为原始协议。
当此协议被更多的人采用,不断的增加、改进、维护、完善。最终形成一个稳定的、完整的文件传输协议,被广泛应用于各种文件传输过程中。该协议就成为一个标准协议。最早的ftp协议就是由此衍生而来。
典型协议:
应用层:
常见的协议有HTTP协议,FTP协议。
HTTP超文本传输协议(Hyper Text Transfer Protocol)是互联网上应用最为广泛的一种网络协议。
FTP文件传输协议(File Transfer Protocol)
传输层:
常见协议有TCP/UDP协议。
TCP传输控制协议(Transmission Control Protocol)是一种面向连接的、可靠的、基于字节流的传输层通信协议。
UDP用户数据报协议(User Datagram Protocol)是OSI参考模型中一种无连接的传输层协议,提供面向事务的简单不可靠信息传送服务。
网络层:
常见协议有IP协议、ICMP协议、IGMP协议。
IP协议是因特网互联协议(Internet Protocol)
ICMP协议是Internet控制报文协议(Internet Control Message Protocol)它是TCP/IP协议族的一个子协议,用于在IP主机、路由器之间传递控制消息。
IGMP协议是 Internet 组管理协议(Internet Group Management Protocol),是因特网协议家族中的一个组播协议。该协议运行在主机和组播路由器之间。
链路层:
常见协议有ARP协议、RARP协议。
ARP协议是正向地址解析协议(Address Resolution Protocol),通过已知的IP,寻找对应主机的MAC地址。
RARP是反向地址转换协议,通过MAC地址确定IP地址。
2>.什么是socket
Socket,英文含义是【插座、插孔】,一般称之为套接字,用于描述IP地址和端口。可以实现不同程序间的数据通信。
Socket起源于Unix,而Unix基本哲学之一就是"一切皆文件",都可以用"打开open –> 读写write/read –> 关闭close"模式来操作。
Socket就是该模式的一个实现,网络的Socket数据传输是一种特殊的I/O,Socket也是一种文件描述符。
Socket也具有一个类似于打开文件的函数调用:Socket(),该函数返回一个整型的Socket描述符,随后的连接建立、数据传输等操作都是通过该Socket实现的。
在TCP/IP协议中,"IP地址+TCP或UDP端口号"唯一标识网络通讯中的一个进程。"IP地址+端口号"就对应一个socket。
欲建立连接的两个进程各自有一个socket来标识,那么这两个socket组成的socket pair就唯一标识一个连接。因此可以用Socket来描述网络连接的一对一关系。
常用的Socket类型有两种:
流式Socket(SOCK_STREAM):。
流式是一种面向连接的Socket,针对于面向连接的TCP服务应用;
数据报式Socket(SOCK_DGRAM):
数据报式Socket是一种无连接的Socket,对应于无连接的UDP服务应用。
温馨提示:
套接字的内核实现较为复杂,不宜在学习初期深入学习,了解到如下结构足矣。
3>.网络应用程序设计模式
C/S模式
传统的网络应用设计模式,客户机(client)/服务器(server)模式。需要在通讯两端各自部署客户机和服务器来完成数据通信。
优点:
1>.客户端位于目标主机上可以保证性能,将数据缓存至客户端本地,从而提高数据传输效率。
2>.一般来说客户端和服务器程序由一个开发团队创作,所以他们之间所采用的协议相对灵活。可以在标准协议的基础上根据需求裁剪及定制。例如,腾讯所采用的通信协议,即为ftp协议的修改剪裁版。
因此,传统的网络应用程序及较大型的网络应用程序都首选C/S模式进行开发。如知名的网络游戏魔兽世界。3D画面,数据量庞大,使用C/S模式可以提前在本地进行大量数据的缓存处理,从而提高观感。
缺点:
1>.由于客户端和服务器都需要有一个开发团队来完成开发。工作量将成倍提升,开发周期较长。
2>.从用户角度出发,需要将客户端安装至用户主机上,对用户主机的安全性构成威胁。这也是很多用户不愿使用C/S模式应用程序的重要原因。
B/S模式
浏览器(Browser)/服务器(Server)模式。只需在一端部署服务器,而另外一端使用每台PC都默认配置的浏览器即可完成数据的传输。
优点:
1>.B/S模式相比C/S模式而言,由于它没有独立的客户端,使用标准浏览器作为客户端,其工作开发量较小。只需开发服务器端即可。
2>.另外由于其采用浏览器显示数据,因此移植性非常好,不受平台限制。如早期的偷菜游戏,在各个平台上都可以完美运行。
缺点:
1>.B/S模式的缺点也较明显。由于使用第三方浏览器,因此网络应用支持受限。
2>.没有客户端放到对方主机上,缓存数据不尽如人意,从而传输数据量受到限制。应用的观感大打折扣。
3>.必须与浏览器一样,采用标准http协议进行通信,协议选择不灵活。
综上所述,在开发过程中,模式的选择由上述各自的特点决定。根据实际需求选择应用程序设计模式。
4>.博主推荐阅读
计算机网络基础之网络拓扑介绍:
https://www.cnblogs.com/yinzhengjie/p/11846279.html
计算机网络基础之OSI参考模型(理论上的标准):
https://www.cnblogs.com/yinzhengjie/p/11846473.html
计算机网络基础之网络设备:
https://www.cnblogs.com/yinzhengjie/p/11853809.html
计算机网络基础之TCP/IP 协议栈(事实上的标准):
https://www.cnblogs.com/yinzhengjie/p/11854107.html
计算机网络基础之IP地址详解:
https://www.cnblogs.com/yinzhengjie/p/11854562.html
二.TCP的socket编程实战案例
1>.简单C/S模型通信
package main
import ("fmt"
"net")
func main() {/**
使用Listen函数创建监听socket,其函数签名如下:
func Listen(network, address string) (Listener, error)
以下是对函数签名的参数说明:
network:
指定服务端socket的协议,如tcp/udp,注意是小写字母哟~
address:
指定服务端监听的IP地址和端口号,如果不指定地址默认监听当前服务器所有IP地址哟~*/socket, err := net.Listen("tcp", "127.0.0.1:8888")if err !=nil {
fmt.Println("开启监听失败,错误原因:", err)return}
defer socket.Close()
fmt.Println("开启监听...")for{/**
等待客户端连接请求*/conn, err :=socket.Accept()if err !=nil {
fmt.Println("建立链接失败,错误原因:", err)return}
defer conn.Close()
fmt.Println("建立链接成功,客户端地址是:", conn.RemoteAddr())/**
接收客户端数据*/buf := make([]byte, 1024)
conn.Read(buf)
fmt.Printf("读取到客户端的数据为: %s\n", string(buf))/**
发送数据给客户端*/tmp := "Blog地址:[https://www.cnblogs.com/yinzhengjie/]"conn.Write([]byte(tmp))
}
}
简单版本服务端代码
package main
import ("fmt"
"net")
func main() {/**
使用Dial函数链接服务端,其函数签名如下所示:
func Dial(network, address string) (Conn, error)
以下是对函数签名的各参数说明:
network:
指定客户端socket的协议,如tcp/udp,该协议应该和需要链接服务端的协议一致哟~
address:
指定客户端需要链接服务端的socket信息,即指定服务端的IP地址和端口*/conn, err := net.Dial("tcp", "127.0.0.1:8888")if err !=nil {
fmt.Println("连接服务端出错,错误原因:", err)return}
defer conn.Close()
fmt.Println("与服务端连接建立成功...")/**
给服务端发送数据*/conn.Write([]byte("服务端,请问博客地址的URL是多少呢?"))/**
获取服务器的应答*/
var buf = make([]byte, 1024)
conn.Read(buf)
fmt.Printf("从服务端获取到的数据为:%s\n", string(buf))
}
简单版本客户端代码
package main
import ("fmt"
"net"
"strconv")
func main() {/**
使用Listen函数创建监听socket,其函数签名如下:
func Listen(network, address string) (Listener, error)
以下是对函数签名的参数说明:
network:
指定服务端socket的协议,如tcp/udp,注意是小写字母哟~
address:
指定服务端监听的IP地址和端口号,如果不指定地址默认监听当前服务器所有IP地址哟~*/socket, err := net.Listen("tcp", "127.0.0.1:8888")if err !=nil {
fmt.Println("开启监听失败,错误原因:", err)return}
defer socket.Close()
fmt.Println("开启监听...")for{/**
等待客户端连接请求*/conn, err :=socket.Accept()if err !=nil {
fmt.Println("建立链接失败,错误原因:", err)return}
defer conn.Close()
fmt.Println("建立链接成功,客户端地址是:", conn.RemoteAddr())/**
分两次接收客户端数据:
第一次最终接收数据的长度;
第二次根据第一次接受的长度,创建容量大小;*/tmp := make([]byte, 2)
conn.Read(tmp)
dataLength, err := strconv.Atoi(string(tmp)) //把字节切片转换成整型
if err !=nil {
fmt.Println("获取数据长度失败:", err)return}
fmt.Println("获取到的数据长度是:", dataLength)
conn.Write([]byte("已获取到数据长度"))/**
开始读取数据*/buf := make([]byte, dataLength)
conn.Read(buf)
fmt.Printf("读取到客户端的数据为: %s\n", string(buf))/**
发送数据给客户端*/data := "Blog地址:[https://www.cnblogs.com/yinzhengjie/]"conn.Write([]byte(data))
}
}
简单版本服务端代码(优化版)
package main
import ("fmt"
"net"
"strconv")
func main() {/**
使用Dial函数链接服务端,其函数签名如下所示:
func Dial(network, address string) (Conn, error)
以下是对函数签名的各参数说明:
network:
指定客户端socket的协议,如tcp/udp,该协议应该和需要链接服务端的协议一致哟~
address:
指定客户端需要链接服务端的socket信息,即指定服务端的IP地址和端口*/conn, err := net.Dial("tcp", "127.0.0.1:8888")if err !=nil {
fmt.Println("连接服务端出错,错误原因:", err)return}
defer conn.Close()
fmt.Println("与服务端连接建立成功...")/**
定义需要发送的数据,第一次给服务端发送要发的长度*/data := []byte("服务端,请问博客地址的URL是多少呢?")
lenData :=len(data)/**
给服务端发送数据的长度*/conn.Write([]byte(strconv.Itoa(lenData)))/**
获取服务器的应答*/
var buf = make([]byte, 1024)
conn.Read(buf)
fmt.Printf("从服务端获取到的数据为:%s\n", string(buf))/**
第二次给服务器发送数据*/conn.Write(data)
conn.Read(buf)
fmt.Printf("获取到的数据为:%s\n", string(buf))
}
简单版本客户端代码(优化版)
2>.并发C/S模型通信
package main
import ("fmt"
"net"
"strings")
func HandleConn(conn net.Conn) {//函数调用完毕,自动关闭conn
defer conn.Close()//获取客户端的网络地址信息
addr :=conn.RemoteAddr().String()
fmt.Println(addr,"conncet sucessful")
buf := make([]byte, 2048)for{//读取用户数据
n, err :=conn.Read(buf)if err !=nil {
fmt.Println("err =", err)return}
fmt.Printf("[%s]: %s\n", addr, string(buf[:n]))
fmt.Println("len =", len(string(buf[:n])))//if "exit" == string(buf[:n-1]) {//nc测试,发送时,只有 \n
if "exit" == string(buf[:n-2]) { //自己写的客户端测试, 发送时,多了2个字符, "\r\n"
fmt.Println(addr, "exit")return}//把数据转换为大写,再给用户发送
conn.Write([]byte(strings.ToUpper(string(buf[:n]))))
}
}
func main() {/**
使用Listen函数创建监听socket,其函数签名如下:
func Listen(network, address string) (Listener, error)
以下是对函数签名的参数说明:
network:
指定服务端socket的协议,如tcp/udp,注意是小写字母哟~
address:
指定服务端监听的IP地址和端口号,如果不指定地址默认监听当前服务器所有IP地址哟~*/socket, err := net.Listen("tcp", "127.0.0.1:8888")if err !=nil {
fmt.Println("开启监听失败,错误原因:", err)return}
defer socket.Close()
fmt.Println("开启监听...")//接收多个用户
for{/**
等待客户端连接请求*/conn, err :=socket.Accept()if err !=nil {
fmt.Println("建立链接失败,错误原因:", err)return}//处理用户请求, 新建一个go程
go HandleConn(conn)
}
}
服务端代码
package main
import ("fmt"
"net"
"strconv")
func main() {/**
使用Dial函数链接服务端,其函数签名如下所示:
func Dial(network, address string) (Conn, error)
以下是对函数签名的各参数说明:
network:
指定客户端socket的协议,如tcp/udp,该协议应该和需要链接服务端的协议一致哟~
address:
指定客户端需要链接服务端的socket信息,即指定服务端的IP地址和端口*/conn, err := net.Dial("tcp", "127.0.0.1:8888")if err !=nil {
fmt.Println("连接服务端出错,错误原因:", err)return}
defer conn.Close()
fmt.Println("与服务端连接建立成功...")/**
定义需要发送的数据,第一次给服务端发送要发的长度*/data := []byte("服务端,请问博客地址的URL是多少呢?")
lenData :=len(data)/**
给服务端发送数据的长度*/conn.Write([]byte(strconv.Itoa(lenData)))/**
获取服务器的应答*/
var buf = make([]byte, 1024)
conn.Read(buf)
fmt.Printf("从服务端获取到的数据为:%s\n", string(buf))/**
第二次给服务器发送数据*/conn.Write(data)
conn.Read(buf)
fmt.Printf("获取到的数据为:%s\n", string(buf))
}
客户端代码
三.UDP的socket编程实战案例
1>.UDP与TCP的差异概述
TCP和UDP的主要区别如下:
1>.TCP是面向连接,UDP是面向无连接
TCP在建立/端口连接时分别要进行三次握手/四次断开,所以我们说TCP是可靠的连接,而说UDP是不可靠的连接;
2>.TCP是流式传输,可能会出现"粘包"问题,UDP是数据报传输,UDP可能会出现"丢包"问题
"粘包"问题可以通过发送数据包的长度解决
"丢包"问题可以通过每一个数据报添加标识位
3>.TCP要求系统资源较多,UDP要求系统资源较少
TCP需要创建连接再进行通信,所以效率要比UDP慢
4>.TCP程序结构较复杂,UDP程序结构较简单
5>.TCP可以保证数据的准确性,而UDP则不保证数据的准确性
应用场景:
TCP的应用场景:
比如文件传输,重要数据传输等。
UDP的应用常见:
比如打电话,直播等.
2>.简单C/S模型通信
package main
import ("fmt"
"net")
func main() {/**
创建监听的地址,并且指定udp协议*/udp_addr, err := net.ResolveUDPAddr("udp", "127.0.0.1:9999")if err !=nil {
fmt.Println("获取监听地址失败,错误原因:", err)return}/**
创建数据通信socket*/conn, err := net.ListenUDP("udp", udp_addr)if err !=nil {
fmt.Println("开启UDP监听失败,错误原因:", err)return}
defer conn.Close()
fmt.Println("开启监听...")
buf := make([]byte, 1024)/**
通过ReadFromUDP可以读取数据,可以返回如下三个参数:
dataLength:
数据的长度
raddr:
远程的客户端地址
err:
错误信息*/dataLength, raddr, err :=conn.ReadFromUDP(buf)if err !=nil {
fmt.Println("获取客户端传递数据失败,错误原因:", err)return}
fmt.Println("获取到客户端的数据为:", string(buf[:dataLength]))/**
写回数据*/conn.WriteToUDP([]byte("服务端已经收到数据啦~"), raddr)
}
简单版本服务端代码
package main
import ("fmt"
"net")
func main() {/**
使用Dial函数链接服务端,其函数签名如下所示:
func Dial(network, address string) (Conn, error)
以下是对函数签名的各参数说明:
network:
指定客户端socket的协议,如tcp/udp,该协议应该和需要链接服务端的协议一致哟~
address:
指定客户端需要链接服务端的socket信息,即指定服务端的IP地址和端口*/conn, err := net.Dial("udp", "127.0.0.1:9999")if err !=nil {
fmt.Println("连接服务端出错,错误原因:", err)return}
defer conn.Close()
fmt.Println("与服务端连接建立成功...")/**
给服务端发送数据*/conn.Write([]byte("Hi,My name is Jason Yin."))/**
读取服务端返回的数据*/tmp := make([]byte, 1024)
n, _ :=conn.Read(tmp)
fmt.Println("获取到服务器返回的数据为:", string(tmp[:n]))
}
简单版本客户端代码