网络通信标准---网络协议
互联网协议--osi七层协议
五层协议:应用层:应用层、表示层、会话层
传输层:传输层
网络层:网络层
数据链路层:数据链路层
物理层:物理层
物理层就是用来发送电信号的
数据链路层跑协议,分组标准。 ethernet以太网协议,规定电信号如何分组
电信号拿来后是一堆数据,只要规定了怎么分组才能拿到正确数据
ethernet规定
一组电信号构成一个数据报,叫做帧
每一数据帧分成:报头head和数据data两部分
但凡接入互联网必须要有网卡
每个网卡出厂都有自己的MAC地址 前六位厂商地址后几位自己地址
MAC是全世界独一无二的
以通讯角度来说,以太网协议有物理层和数据链路层已经可以实现通讯了
以太网有广播的工作方式
广播包只能在局域网里面传,要不数据量太大会造成网络的故障
网络层---ip协议 ipv4/ipv6
ip数据包也分成head和data部分 ip头的意思和以太网差不多
ip地址和ip子网掩码中间的运算得出一个地址,是网段地址
按位与运算 两个同时是1才是1 二进制的ip地址和子网掩码进行计算
按照这个网段地址找到范围,然后再找到这个机器
ipv4/ipv6网关是用来帮助跨过子网的关口
跨子网,要用网关和网关进行通讯
用以太网的广播方式将数据交给网关,但是得先获得网关的MAC地址
但是我们知道的只有网关的IP地址
所以需要ARP协议,局域网内,用IP地址解析成MAC地址。这样用网关的IP地址获取MAC地址
才能将数据交给网关
如果发送端主机发送了目标MAC地址为FF.FF.FF.FF.FF.FF那么它就知道发送端需要获取MAC
但是接受数据的主机不知道是不是向它要MAC地址,所以还需要发送一个IP地址,让他知道是自己需要的
ARP协议里面的
同一子网地址内都用以太网的广播方式
IP地址和MAC地址可以确定一个机器在什么位置
-------------------------------------------------------------------------------------------------------------------------------------------
传输层跑两种协议:tcp协议和udp协议得到端口 确定要传输的是这个机器的哪个软件
ip加端口确定了全世界独一无二的软件
软件用机器的哪个端口,65536个端口前1024是操作系统用,后面全是自己的软件用
应用层:httpp ftp mai 协议 可以自己定协议
数据前一般都要加头 然后是数据
DHCP :自动分配IP地址
DNS:域名服务 将网址域名地址解析成IP地址
tcp协议:
建立双向连接,用来进行数据的交互相互发送。
工作方式:c传给s一个请求,然后s再发送一个确认和一个请求,然后c再发送一个确认
所以一共发送了三次,三次握手
为了确认是否以我发送的那个数据,你俩进行回应,所以我的数据要加一个序列
如果你发送回来的时候我的序列+1证明是用我的数据进行回应
tcp协议又称为可靠协议,发送完包一定要等待对方确认收到了我的包进行了回应,才算发成功
如果没有回应过一段时间再发一次,所以是可靠协议。
可靠在发送完数据并不清空内存,直到对方回应确认收到后才收到
udp协议不用建立连接,被称为不可靠协议,发送完数据就清空内存,不等对方的确认信息。
tcp有断开连接,udp没有
谁最后确认数据收到谁先断,四次挥手断链接
------------------------------------------------------------------------------------------------------------------------------------
socket编程:
应用层与运输层之间多了一个socket抽象层
将传输协议都封装好直接用
基于应用层与传输层之间的抽象层
用基于网络的套接字:AF——INET
服务器端:先初始化socket(),然后与端口绑定bind()服务端的固定地址,对端口进行监听listen()看谁来链接了
accept()阻塞直到有客户端链接 和connect()组成一队进行链接
import socket
socket.socket(socket.AF_INET,socket.SOCK_STREAM)
服务端:
import socket
#买手机
phone=socket.socket(socket.AF_INET,socket.SOCK_STREAM) 打开了这个服务端最后要关闭
#绑定手机卡
phone.bind((‘127.0.0.1‘,8080))#127.0.0.1指的就是自己这台机器的IP地址
#开机
phone.listen(5)#5代表最大挂起的链接数
#等电话链接
conn,addr=phone.accept() 等待建立链接accept()等待建立
print(‘客户端IP:%s,客户端端口:%s‘%(addr[0],addr[1]))
data=conn.recv(1024) #最大收1024 recv()接受数据,括号内最大接收量
print(data.decode(‘utf-8‘))
conn.send(data.upper()) send()发送数据
#6、挂电话
conn.close() 建立的链接要关闭
#7.关机
phone.close()
客户端:
import socket
#买手机
phone=socket.socket(socket.AF_INET,socket.SOCK_STREAM) 一样打开这个端口
#绑定手机卡 客户端没必要绑定所以不用了
#开机 也不用监听了
phone.connect((‘127.0.0.1‘,8080)) 主动建立链接,前面IP地址后面是端口
phone.send(‘hello‘.encode(‘utf-8‘)) 发送数据
data=phone.recv(1024)
print(data.decode(‘utf-8‘))
#关机
phone.close() 同样要关机
持续性数据的交互,还有排队的进行链接:
服务端:
import socket
#买手机
phone=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
#绑定手机卡
phone.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)
phone.bind((‘192.168.11.52‘,8080))#127.0.0.1指的就是自己这台机器的IP地址
#开机
phone.listen(5)#5代表最大挂起的链接数
#等电话链接
print(‘starting...‘)
while True: #链接循环
conn,addr=phone.accept()
print(‘客户端IP:%s,客户端端口:%s‘%(addr[0],addr[1]))
while True: #通信循环
try: #用到try是因为有时候会发生但无法预知的错误
data=conn.recv(1024) #最大收1024
if not data:break #针对linux
print(data.decode(‘utf-8‘))
conn.send(data.upper())
except ConnectionResetError:
break
#6、挂电话
conn.close() #断开连接
#7.关机
phone.close()
客户端:
import socket
#买手机
phone=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
#绑定手机卡 客户端没必要绑定所以不用了
#开机 也不用监听了
#等电话链接
phone.connect((‘192.168.11.52‘,8080))
while True:
msg=input(‘请输入:‘).strip()
phone.send(msg.encode(‘utf-8‘))
data=phone.recv(1024)
print(data.decode(‘utf-8‘))
#关机
phone.close()
客户端向服务端发送空消息的话 服务端的接收会卡住