网络通信协议
本节主要内容:
1.osi七层模型
应 表 会 传 网 数 物
2.socket
tcp和udp协议下的socket
传输层: 端口号
Tcp协议:面向连接,数据可靠,传输效率低,面向字节
Udp协议:面向无连接,数据不可靠,传输效率高,面向报文
一、osi七层模型
互联网的核心就是由一堆协议组成的,协议就是标准,标准就是大家都认可的,所有人都按照这个来,这样大家都能够互相了解,互相深入。比如全世界通用语言是英语。
五层通信流程:
二、socket
结合上面五层通信流程来看,socket在哪一层呢??
1.socket在内的五层通讯流程:
Socket又称为套接字,它是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口。在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面,对用户来说,一组简单的接口就是全部,让Socket去组织数据,以符合指定的协议。当我们使用不同的协议进行通信时就得使用不同的接口,还得处理不同协议的各种细节,这就增加了开发的难度,软件也不易于扩展(就像我们开发一套公司管理系统一样,报账、会议预定、请假等功能不需要单独写系统,而是一个系统上多个功能接口,不需要知道每个功能如何去实现的)。于是UNIX BSD就发明了socket这种东西,socket屏蔽了各个协议的通信细节,使得程序员无需关注协议本身,直接使用socket提供的接口来进行互联的不同主机间的进程的通信。这就好比操作系统给我们提供了使用底层硬件功能的系统调用,通过系统调用我们可以方便的使用磁盘(文件操作),使用内存,而无需自己去进行磁盘读写,内存管理。socket其实也是一样的东西,就是提供了tcp/ip协议的抽象,对外提供了一套接口,同过这个接口就可以统一、方便的使用tcp/ip协议的功能了。
其实站在你的角度上看,socket就是一个模块。我们通过调用模块中已经实现的方法建立两个进程之间的连接和通信。也有人将socket说成ip+port,因为ip是用来标识互联网中的一台主机的位置,而port是用来标识这台机器上的一个应用程序。 所以我们只要确立了ip和port就能找到一个应用程序,并且使用socket模块来与之通信。
2.套接字socket的发展史及分类
套接字起源于 20 世纪 70 年代加利福尼亚大学伯克利分校版本的 Unix,即人们所说的 BSD Unix。 因此,有时人们也把套接字称为“伯克利套接字”或“BSD 套接字”。一开始,套接字被设计用在同 一台主机上多个应用程序之间的通讯。这也被称进程间通讯,或 IPC。套接字有两种(或者称为有两个种族),分别是基于文件型的和基于网络型的。
基于文件类型的套接字家族
套接字家族的名字:AF_UNIX
unix一切皆文件,基于文件的套接字调用的就是底层的文件系统来取数据,两个套接字进程运行在同一机器,可以通过访问同一个文件系统间接完成通信。
基于网络类型的套接字家族:
套接字家族的名字:AF_INET
(还有AF_INET6被用于ipv6,还有一些其他的地址家族,不过,他们要么是只用于某个平台,要么就是已经被废弃,或者是很少被使用,或者是根本没有实现,所有地址家族中,AF_INET是使用最广泛的一个,python支持很多地址家族,但是由于我们只关心网络编程,所以大部分时候我们只使用AF_INET)
3.基于TCP和UDP两个协议下socket的通讯流程
TCP(Transmission Control Protocol)可靠的、面向连接的协议(eg:打电话)、传输效率低全双工通信(发送缓存&接收缓存)、面向字节流。使用TCP的应用:Web浏览器;文件传输程序。
UDP(User Datagram Protocol)不可靠的、无连接的服务,传输效率高(发送前时延小),一对一、一对多、多对一、多对多、面向报文(数据包),尽最大努力服务,无拥塞控制。使用UDP的应用:域名系统 (DNS);视频流;IP语音(VoIP)。
看下图的对比差异:
接续往下看:
TCP和UDP下的socket差异对比图:
上面的图只是让大家感受一下TCP和UDP协议下,socket工作流程的不同,两者之间的差异是TCP需要连接,UDP不需要。现在是不是有点迷糊!这里面的bind、listen啥的都是什么东西啊。下面我们就分开两者。
三、TCP协议下的socket
基于TCP的socket通讯流程图片:
虽然上图将通讯流程中的大致描述了一下socket各个方法的作用,但是还是要总结一下通讯流程(下面一大段内容)
先从服务器端说起,服务器端先初始化socket,然后与端口绑定(bind()),对接口进行监听(listen()),调用accept阻塞,等待客户端连接。在这时如果有一个客户端初始化一个socket,然后连接服务器(connect),如果连接成功,这时客户端与服务器的连接就建立了。客户端发送数据请求,服务器端接收请求并处理请求,然后把回应数据发给客户端,客户端读取数据,最后关闭连接,一次交互结束。
上代码感受下,需要创建两个文件,文件名我这里为了方便看,我的两个文件名称为tcp_server.py(服务端)和tcp_client.py(客户端),将下面的server端的代码拷贝到tcp_server.py文件中,将下面client端的代码拷贝到tcp_client.py中,然后先运行服务端,再运行客户端。
server端代码实例(如果比喻成打电话)
解决办法:
但是如果你加上了上面的代码之后还是出现这个问题:OSError: [WinError 10013] 以一种访问权限不允许的方式做了一个访问套接字的尝试。那么只能换端口了,因为你的电脑不支持端口重用。
记住一点,用socket进行通信,必须是一收一发对应好。
提一下:网络相关或者需要和电脑上其他程序通信的程序才需要开一个端口。
你会发现,第一个连接的客户端可以和服务端收发消息,但是第二个连接的客户端发消息服务端是收不到的
原因解释:
TCP协议下的socket实现多人通话, listen(3) 要排队等待的有三个客户端:
server端
import socket
server = socket.socket()
server.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)
server_ip = ("127.0.0.1",1026)
server.bind(server_ip)
server.listen(3)
while 1 :
conn, address = server.accept()
while 1 :
from_client = conn.recv(1024)
print("客户端说:",from_client.decode("utf-8"))
if from_client.decode("utf-8").upper() == "Q":
break
a = input("我说:")
conn.send(a.encode("utf-8"))
conn.close()
client端
import socket
client = socket.socket()
client_ip = ("127.0.0.1",1026)
client.connect(client_ip)
while 1 :
a = input("我说:")
client.send(a.encode("utf-8"))
if a.upper() == "Q" :
break
from_server = client.recv(1024)
print("服务端说:",from_server.decode("utf-8"))
client.close()
四、UDP协议下的socket
基于UDP的socket通讯流程:
总结一下UDP下的socket通讯流程
先从服务器端说起。服务器端先初始化Socket ( 创建对象 utp_server = socket.socket ( type = socket.SOCK_DGRAM ) ),然后与端口绑定(bind),recvform接收消息,这个消息有两项,消息内容和对方客户端的地址,然后回复消息时也要带着你收到的这个客户端的地址,发送回去,最后关闭连接
server端代码的实例:(实现多人与服务端通话)
server端
import socket utp_server = socket.socket(type=socket.SOCK_DGRAM) utp_server_ip = ("127.0.0.1",1030) utp_server.bind(utp_server_ip) while 1 : from_client,client_add = utp_server.recvfrom(1024) print("客户端说:",from_client.decode("utf-8"),"客户端的地址和端口:",client_add) a = input("我说:") utp_server.sendto(a.encode("utf-8"),client_add)
client端
import socket utp_client = socket.socket(type=socket.SOCK_DGRAM) server_ip = ("127.0.0.1",1030) while 1 : a = input("我说:") utp_client.sendto(a.encode("utf-8"),server_ip) from_server,server_add = utp_client.recvfrom(1024) print("服务端说:",from_server.decode("utf-8"),"服务端的地址和端口",server_add) if a.upper() == "Q" and from_server.decode("utf-8").upper() == "Q" : break
utp_client.close()
这里是两个简易描述socket各个参数和方法的图,供大家参考:
socket类型:
socket各个方法的解释: