整个计算机网络都是由协议组成。
其实可以把协议类比成我们人类用的语言,我们和老外交流,双方得说一样的语言,就好像遵从同样的“协议”。网络上的两台机器之间要交流,也是如此。
七层网络模型——OSI标准
OSI 七层模型是一个标准,规定了机器(主要是电脑)之间如何通信。因此假如你想要让你的洗碗机和洗衣机通信的话,就需要遵从 OSI 模型,或者至少从 OSI 模型获得启发。这意味着需要遵从分层的通信方式。
关于这七层模型,有一个记忆口诀:
All People Seem To Need Data Processing
翻译成英语是“似乎所有人都需要数据处理”。这一句话由七个英语单词组成,每一个的首字母正好是按第 7 层到第 1 层的首字母的顺序。
记忆 | 解释 |
All | Application |
People | Presentation |
Seem | Session |
To | Transport |
Need | Network |
Date | Datelink |
Process | Physical |
当然,????????标准和实际应用还是有区别的,实际运用起来是如下的五层网络模型。
五层网络模型
OSI | 功能 | TCP/IP协议 |
应用层 | 文件传输、电子邮件、文件服务 | HTTP、FTP、STMTP、DNS、Telmet等 |
传输层 | 提供端对端的接口 | TCP、UDP |
网络层 | 为数据包选择路由 | IP、ICMP等 |
数据链路层 | 传输有地址的帧、错误检测功能 | ARP 等 |
物理层 | 物理媒体 | 1000BASE-SX等 |
比如:
A客户端 – B服务器传输数据:
首先经过应用层,
往下,到达传输层,所有的都会经过传输
往下网络层,路由选择
数据链路层,中继器,路由器
最后到达物理层,网线 或者 无线电波
由下往上,数据组包,组好数据–>字符串或者JSON数据,
我们用到最多的是应用层协议,
根据IP地址到 Web服务器–>操作系统–>80端口–>Ngix–>UWSGI
解析地址–DNS,通过域名查询IP协议。
浏览器实现了HTTP协议
聊天工具与HTTP协议不一样,如果直接与底层协议打交道,编程复杂
操作系统给我们提供了SOCKET,一种API,方便直接跟传输层打交道。
实现自己的应用功能,HTTP协议单向,Socket编程的意义。Socket本身并不是协议,是工具,用于连接应用和TCP/UDP打交道,从而实现自己的协议。
Client和Server实现通信
TCP服务器和客户端连接图如下:
Python Socket编程
简易通信:
- 创建一个服务器
server_socket.py
,绑定本地IP,端口9999,然后进行服务监听,然后等待客户端发送数据,并打印从客户端接收到的数据。
# server_socket.py import socket # 创建套接字 server = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 绑定IP和端口,元组存储地址和端口 server.bind(('0.0.0.0', 9999)) # 服务端监听 server.listen() # accept接受到用户的请求 sock, addr = server.accept() # 接受从客户端发送的数据,一次获取1k的数据 data = sock.recv(1024) # 打印接收的数据 print(data.decode('utf8')) # 关闭服务器连接 server.close()
- 创建客户端
client_socket.py
,向本地IP,9999端口发起连接
# client_socket.py import socket # 创建套接字 client = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 要连接的服务器 client.connect(('127.0.0.1', 9999)) # 要发送的数据 client.send('Hello, Server'.encode('utf8')) # 关闭连接 client.close()
先运行服务器程序,然后允许客户端程序,再回到服务器端控制台,可以看到如下输出:
双向通信:
# socket_server.py import socket # 创建套接字 server = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 绑定IP和端口,元组存储地址和端口 server.bind(('0.0.0.0', 8888)) # 服务端监听 server.listen() # accept接受到用户的请求 sock, addr = server.accept() # 循环发送与接收消息 while True: # 接受从客户端发送的数据,一次获取1k的数据 data = sock.recv(1024) # 打印客户端接收的数据 print('客户端:', data.decode('utf8')) # 服务器发送给客户端的数据 re_data = input('服务器:') sock.send(re_data.encode('utf8'))
# client_socket.py import socket # 创建套接字 client = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 要连接的服务器 client.connect(('127.0.0.1', 8888)) while True: # 要发送的数据 re_data = input('客户端:') client.send(re_data.encode('utf8')) # 要接收的数据 data = client.recv(1024) print('服务端:', data.decode('utf8'))
我们还是先运行服务器代码socket_server.py
,接着运行客户端client_socket.py
- 客户端给服务器发送:你好,服务器。
- 在服务器端收到信息后,回复一个:Hello,客户端
- 客户端再给服务端:一起来聊天吧
演示如下:
通过这个过程,实现了单个客户端与服务器的通信,如果说一个服务器程序想同时跟多个客户端交互,就需要利用到多线程编程,这里先不讲。
Socket编程是网络编程的重要组成部分,掌握异步IO和协程的前提,关于Socket编程还有更多内容等着学习。