一、网络编程概念
网络的功能:数据传输
OSI七层模型------->网络通信工作流程的标准化模型
应用层:提供用户服务,具体功能由程序而定
表示层:提供数据的压缩、优化、加密
会话层:建立应用级的连接,选择传输服务
传输层:提供不同的传输服务,流量控制
网络层:路由选择,网络互连
链路层:进行数据转换,具体消息的发送,链路连接
物理层:物理硬件,接口设定,网卡路由交换机等
OSI七层模型的优点:
1、将工作流程标准化
2、降低了模块之间的耦合度,使每一部分可以单独开发,单独工作
四层模型
应用层:集合 应用层,表示层,回话层
传输层
网络层
物理链路层
五层模型(tcp/Ip)
应用层:集合 应用层,表示层,回话层
传输层
网络层
链路层
物理层
数据传输流程
- 发送端有应用层到物理层逐层添加信息头(首部)最终在物理层发送
- 中间经过节点(交换机,路由器等)转发,发送到接受端
- 在接收端根据发送端的每个信息头进行解析,最终消息到应用层展示给用户
网络协议:
-
在网络通信中双方都遵循的规定,包括建立什么样的网络结 构,消息结构,标识代表什么等
-
应用层:TFTP HTTP DNS SMTP
-
传输层:TCP UDP
-
网络层:IP
-
物理层:IEEE
互联网相关概念
-
网络主机:在网络中标识一台计算机 HOST
-
本地使用:‘localhost’、‘127.0.0.1’
-
网络使用:‘0.0.0.0’ 标识当前计算机可用网卡
-
ifconfig:查看ip信息
-
获取计算机名称
In [2]:socket.gethostname()
Out[2]: ‘tedu’ -
通过计算机名获取地址
In [4]: socket.gethostbyname(‘localhost’)
Out[4]: ‘127.0.0.1’ -
IP地址
网络上确定一台主机位置的地址
IPv4:点分十进制 192.168.1.2 0~255
IPv6:128位 -
特殊IP地址
127.0.0.1本地测试ip
0.0.0.0使用本机可用ip
192.168.1.0表示网段ip
192.168.1.1网关ip
192.168.1.255广播地址 -
通过地址获取主机网络信息
In [7]: socket.gethostbyaddr(‘www.baidu.com’)
Out[7]: (‘127.0.0.1’, [], [‘119.75.216.20’])
主机名 别名 网络地址 -
点分十进制地址转换为二进制
In [9]: socket.inet_aton(‘192.168.1.2’)
Out[9]: b’\xc0\xa8\x01\x02’ -
二进制地址转换为点分十进制
In [10]: socket.inet_ntoa(b’\xc0\xa8\x01\x02’)
Out[10]: ‘192.168.1.2’ -
域名:网络服务器地址的名称
1、方便记忆
2、名称表达一定的含义 -
网络端口号
端口是网络地址的一部分,用于区分一个网络主机上的网络 应用
*在一个操作系统中不同的网络应用监听不同的端口号
取值范围1~65535
1~255 一些众所周知的通用端口
256~1023 系统应用端口
1024~65535 自用端口
建议使用大于10000的端口 -
获取一个应用的端口信息
In [20]: socket.getservbyname(‘mysql’)
Out[20]: 3306 -
网络字节序:
数据在网络中的传输格式
传输层服务
-
面向连接的传输服务(tcp)
基于tcp协议的数据传输- 传输特征:
提供可靠的数据传输,可靠性指数据传输过程中无丢 失,无失序,无差错,无重复。 - 实现手段:
在数据的传输断开前都需要进行传输和断开的确认 - 三次握手:tcp传输在数据传输前建立连接的过程
1、客户端向服务器发送连接请求
2、服务器收到请求后,回复确认消息,表示允许访问
3、客户端收到服务器回复,进行最终标志发送确认连接
- 四次挥手:tcp传输在连接断开前进行断开确认的过程
1、主动方发送报文告知被动方要断开连接
2、被动方收到请求后立即返回报文告知已经准备断开
3、被动方准备就绪后再次发送报文告知可以断开主动方发送消息,确认最终断开 - 应用情况:适用于传输较大的文件,网络情况良好,需要保证传输可靠性的情况
- 比如:网页的获取,文件下载,邮件传输,登录注册
- 传输特征:
-
面向无连接的传输服务(udp)
基于udp协议的传输- 传输特点:不保证传输的可靠性,传输过程没有连接和断 开的流程,数据收发自由。
- 使用情况:网络情况较差,对传输可靠性要求不高,需要 提升传输效率。不便连接,需要灵活收发消息
- 比如:网络视频,群聊,广播发送
socket 套接字编程
-
学习目标:根据socket模块提供的接口函数,进行组合使用完成基于 tcp或者udp的网络编程。
-
套接字:完成上述目标的一种编程手段,编程方案
-
实质: 两个网络各自通信连接中的端点
-
作用:完成两个应用程序之间的数据传输
-
套接字分类:
流式套接字(SOCK_STREAM):传输层基于tcp协议的套接 字编程方案
数据报套接字(SOCK_DGRAM):传输层基于udp协议的套 接字编程方案
底层套接字(SOCK_RAM):访问底层协议的套接字编程 -
面向连接的传输–>tcp–> 可靠的–> 流式套接字
-
面向无连接的传输–> udp–> 不可靠–> 数据报套接字
-
tcp(流式)套接字服务端编程
import socket # 1、创建套接字 sockfd=socket.socket(socket_family=AF_INET,Socket_type=SOCK_STREAM,proto=0) # 功能:创建套接字 # 参数:socket_family:选择地址族类型 AF_INET (ipv4) # socket_type:套接字类型 SOCK_STREAM 流式 # SOCK_DGRAM 数据报 # proto:选择自协议类型 通常为0 默认值为0 # 返回值:返回套接字对象 # 2、绑定地址 :绑定套接字到一个IP地址和一个端口上 sockfd.bind(addr) # 功能:绑定ip地址 # 参数:元组 (ip,port) # localhost 可以被本机用 127.0.0.1 # 127.0.0.1 同上 # 192.168.205.127(本机ip) 可以被所有人用 192.168.205.127 访问 # 0.0.0.0 可以被别人用 192.168.205.127 访问 # 也可以被自己用127.0.0.1访问 # 3、设置监听套接字 sockfd.listen(n) # 功能:将套接字设置为监听套接字,创建监听队列 # 参数:n表示监听队列大小 # *一个监听套接字可以连接多个客户端套接字 # 4、等待处理客户端的连接请求 connfd,addr = sockfd.accept() # 功能:阻塞等待处理客户端连接 # 返回值:connfd 客户端连接套接字 addr 连接的客户端地址 # *阻塞函数:程序运行过程中遇到阻塞函数则暂停运行,直到达#到某种阻塞条件达成再继续运行。 # 5、消息收发 connfd.recv(buffersize) # 功能:接收对应客户端消息 # 参数:一次最多接收多少字节 # 返回值:接收到的内容 # *如果没有消息则会阻塞 n = connfd.send(data) # 功能:发送消息给对应客户端 # 参数:要发送的内容,必须为bytes格式 # 返回值:返回实际发送消息的大小 # 6、关闭套接字 sockfd.close() # 功能:关闭套接字
# server示例 from socket import * # 创建套接字 sockfd = socket(AF_INET, SOCK_STREAM) # 绑定地址 sockfd.bind(('0.0.0.0', 10017)) # 设置监听套接字 sockfd.listen(3) # 等待接收连接 print("Waiting for connect....") connfd, addr = sockfd.accept() print("Connect from", addr) # 收发消息 while True: data = connfd.recv(1024).decode() if data == '##': break print(data) # 发 n = connfd.send(b"Receive your message") print("发送了%s字节" % n) # 关闭套接字 connfd.close() sockfd.close()
-
tcp(流式)套接字客户端编程
-
创建过程
1、创建套接字
*必须相同类型套接字才能通信
2、建立连接
sockfd.connect(server_addr)
功能:建立连接
参数:元组,服务端地址
3、消息收发
*消息收发与服务端配合,避免两端都出现recv堵塞
4、关闭套接字 -
套接字传输注意事项
1、监听套接字存在客户端即可发起连接,但是最终连接的处理需要accept进行处理
2、如果连接的另一端退出,则recv会立即返回空子串不再阻塞
3、当连接的另一端退出时,在试图send发送就会产生BrokenPipeError# client示例 from socket import * # 创建套接字 sockfd = socket(AF_INET, SOCK_STREAM) # 发起连接 server_addr = ("127.0.0.1", 10017) sockfd.connect(server_addr) # 发送接收 while True: data = input("发送>>") sockfd.send(data.encode()) if data == '##': break data = sockfd.recv(1024) print("接收到:", data.decode()) # 关闭 sockfd.close()