网络编程之 TCP-UDP的详细介绍
一、TCP协议
1. TCP协议的特点
1.TCP是面向连接的运输层协议。这就意味着,在使用该协议之前,必须建立TCP连接。在传输数据完毕后,必须释放已经建立的TCP连接。
2.每一条TCP连接只能有两个端点,每一条TCP连接只能是点对点的(一对一)。
3.TCP提供可靠交付的服务。通过TCP连接传送的数据,无差错、不丢失、不重复、并且按序到达。
4.TCP提供全双工通信。TCP允许通信双方的应用进程在任何时候都能发送数据。
TCP连接的两端都设有发送缓存和接收缓存,用来临时存放双向通信的数据。
在发送时,应用程序在把数据传送给TCP的缓存后,就可以做自己的事,而TCP在合适的时候把数据发出去。
在接收时,TCP把收到的数据放入缓存,上层的应用进程在合适的时候读取缓存中的数据。
5.面向字节流。虽然应用程序和TCP的交互试一次一个数据块(注意:大小是不等的),
但TCP把应用程序交下来的数据仅仅看成是一连串的无结构的字节流。TCP并不知道所传送的字节流的含义。
2.三次握手、四次挥手
3.代码实现
3.1 使用TCP协议实现简单通信
# 客户端:
import socket
client = socket.socket()
client.connect(('127.0.0.1', 8080))
client.send(b"this is client!")
data = client.recv(1024)
print(data)
client.close()
# 服务端:
import socket
server = socket.socket() # 实例化sever对象-->买手机
server.bind(('127.0.0.1', 8080)) # 绑定ip和端口-->手机卡
server.listen(5) # 监听 -->打开手机
conn, addr = server.accept() # 等待建立连接
data = conn.recv(1024) # 接收数据 --> 打电话
print(data)
conn.send(b"received data!!")
conn.close() # 关闭连接-->挂电话
server.close() # 关闭服务器 -->关机
3.2 使用TCP协议实现通信循环
# 客户端:
import socket
client = socket.socket()
client.connect(('127.0.0.1', 8080))
while True:
msg = input(">>>")
if msg == 'q':
break
client.send(msg.encode('utf-8'))
data = client.recv(1024)
print(data)
client.close()
# 服务端
import socket
server = socket.socket() # 实例化sever对象-->买手机
server.bind(('127.0.0.1', 8080)) # 绑定ip和端口-->手机卡
server.listen(5) # 监听 -->打开手机
conn, addr = server.accept() # 等待建立连接
while True:
try:
data = conn.recv(1024) # 接收数据 --> 打电话
print(data)
conn.send(b"received data!!")
except ConnectionResetError:
break
conn.close() # 关闭连接-->挂电话
server.close() # 关闭服务器 -->关机
3.3 使用TCP协议实现链接循环
# 客户端
import socket
client = socket.socket()
client.connect(('127.0.0.1', 8080))
while True:
msg = input(">>>")
if msg == 'q':
break
client.send(msg.encode('utf-8'))
data = client.recv(1024)
print(data)
client.close()
# 服务端
import socket
server = socket.socket() # 实例化sever对象-->买手机
server.bind(('127.0.0.1', 8080)) # 绑定ip和端口-->手机卡
server.listen(5) # 监听 -->打开手机 (半连接池)
while True:
conn, addr = server.accept() # 等待建立连接
while True:
try:
data = conn.recv(1024) # 接收数据 --> 打电话
print(data.decode('utf-8'))
conn.send(b"received data!!")
except ConnectionResetError:
break
conn.close() # 关闭连接-->挂电话
server.close() # 关闭服务器 -->关机
3.4 使用TCP协议实现粘包问题
# 客户端
import socket
import struct
import json
client = socket.socket()
client.connect(('127.0.0.1', 8090))
while True:
msg = input(">>>").encode('utf-8')
if len(msg) == 0:
continue
client.send(msg)
header = client.recv(4)
# 对头进行解包,获取真实数据的长度
head_len = struct.unpack('i', header)[0]
head_dic = json.loads(client.recv(head_len).decode('utf-8'))
print(head_dic)
# 对需要接收的数据进行循环接收
total_size = head_dic['len']
recv_size = 0
res = b''
while recv_size < total_size:
data = client.recv(1024)
res += data
recv_size += len(data)
print(res.decode("gbk"))
# 服务端
import socket
import subprocess
import struct
import json
server = socket.socket() # 实例化sever对象-->买手机
server.bind(('127.0.0.1', 8090)) # 绑定ip和端口-->手机卡
server.listen(5) # 监听 -->打开手机 (半连接池)
while True:
conn, addr = server.accept() # 等待建立连接
while True:
try:
data = conn.recv(1024).decode('utf-8') # 接收数据 --> 打电话
if len(data) == 0: break # 针对mac和Linux,客户端异常断开反复收空的情况
# 生成一个能够读取输出、错误流的对象
obj = subprocess.Popen(data, shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
stdout = obj.stdout.read() # 获得输出流
stderr = obj.stderr.read() # 获取错误流
# 打印输出流+输入流的长度
print(len(stdout+stderr))
# 制作一个文件头
header_dic = {
'filename': 'cls.avi',
'len': len(stderr+stdout) # 错误流 或者 输出流 的长度
}
# 序列化文件头字典,得到文件头的二进制文件
header_bytes = json.dumps(header_dic).encode('utf-8')
# 制作报头
header = struct.pack('i', len(header_bytes)) # 将要发送给客户端的数据打包成固定4个字节
conn.send(header)
conn.send(header_bytes)
conn.send(stdout+stderr)
except ConnectionResetError:
break
conn.close() # 关闭连接-->挂电话
server.close() # 关闭服务器 -->关机
3.5 解决粘包问题用到的 struct
1.说明
struct模块中最重要的三个函数是pack(), unpack(), calcsize()
pack(fmt, v1, v2, ...) 按照给定的格式(fmt),把数据封装成字符串(实际上是类似于c结构体的字节流)
unpack(fmt, string) 按照给定的格式(fmt)解析字节流string,返回解析出来的tuple
calcsize(fmt) 计算给定的格式(fmt)占用多少字节的内存
struct中支持的格式如下表:
网址 : https://www.cnblogs.com/gala/archive/2011/09/22/2184801.html
2.代码实现(例子)
import struct
data = 'datafa'
# 服务端
res = struct.pack('i', len(data))
print(res) # b'\x06\x00\x00\x00'
print(len(res)) # 4
3.6 解决粘包问题用到的 subprocess
1.说明
subprocess 模块允许你生成新的进程,连接它们的输入、输出、错误管道,并且获取它们的返回码。
class subprocess.Popen(
args, bufsize=-1, executable=None,
stdin=None, stdout=None, stderr=None, preexec_fn=None,
close_fds=True, shell=False, cwd=None, env=None,
universal_newlines=None, startupinfo=None, creationflags=0,
restore_signals=True, start_new_session=False,
pass_fds=(), *, encoding=None, errors=None, text=None)
2.代码实现(例子):
cmd = r'dir file_list'
obj = subprocess.Popen(
cmd,
shell=True,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
)
stdout = obj.stdout.read() # 来接住stdout流
print('stdout', stdout.decode('gbk')) # 打印进程列表
stderr = obj.stderr.read()
print('stderr', stderr.decode('gbk'))
# 当上面的命令出错时,接收错误流
# stderr 'tasklist1' 不是内部或外部命令,也不是可运行的程序或批处理文件。
# stdout 和 stderr同时存在的时候,有一个接收到了数据另一个就不会停止等待
二、UDP协议
1. UDP协议的特点
1.UDP是无连接的,即发送数据之前不需要建立连接(发送数据结束时也没有连接可释放),因此减少了开销和发送数据之前的时延。
2.UDP使用尽最大努力交付,即不保证可靠交付,因此主机不需要维持复杂的连接状态表。
3.UDP是面向报文的。发送方的UDP对应用程序交下来的报文,在添加首部后就向下交付IP层。UDP对应用层交下来的报文,既不合并,也不拆分,而是保留这些报文的边界。
4.UDP没有拥塞控制,因此网络出现的拥塞不会使源主机的发送速率降低。
5.UDP支持一对一,一对多,多对一和多对多的交互通信。
6.UDP的首部开销小,只有8个字节,比TCP的20个字节的首部要短。
2. 代码实现
2.1 使用UDP协议实现简单通信
# 客户端
import socket
client = socket.socket(type=socket.SOCK_DGRAM)
data = b'this is client'
server_addr = ('127.0.0.1', 8080)
client.sendto(data, server_addr)
data1, addr1 = client.recvfrom(1024)
print(addr1)
print(data1)
# ('127.0.0.1', 8080)
# b'THIS IS CLIENT'
# 服务端
import socket
server = socket.socket(type=socket.SOCK_DGRAM) # udp协议
server.bind(('127.0.0.1', 8080))
while True:
data, addr = server.recvfrom(1024)
print(data)
data = data.decode('utf-8')
data = data.upper()
server.sendto(data.encode('utf-8'), addr)
# b'this is client'
2.2 使用UDP协议实现极简版QQ
# 客户端
import socket
client = socket.socket(type=socket.SOCK_DGRAM)
server_addr = ('127.0.0.1', 8080)
while True:
data = input(">>>:").strip()
client.sendto(data.encode('utf-8'), server_addr)
msg, addr = client.recvfrom(1024)
print(msg.decode('utf-8'))
# 服务端
import socket
server = socket.socket(type=socket.SOCK_DGRAM)
server.bind(('127.0.0.1', 8080))
while True:
data, addr = server.recvfrom(1024)
print(addr)
print(data.decode('utf-8'))
re_msg = input(">>>:")
server.sendto(re_msg.encode('utf-8'), addr)
2.3 socketServer模块可以使TCP通信达到并发效果
# 客户端
import socket
client = socket.socket()
client.connect(('127.0.0.1', 8080))
while True:
msg = input(">>>:")
client.send(msg.encode('utf-8'))
data = client.recv(1024)
print(data.decode('utf-8'))
# 服务端
import socketserver
server_addr = ('127.0.0.1', 8080)
class MySock(socketserver.BaseRequestHandler):
def handle(self):
# 通信循环
while True:
data = self.request.recv(1024) # 接收数据
print(data.decode('utf-8'))
re_msg = input(">>>:")
self.request.send(re_msg.encode('utf-8'))
if __name__ == '__main__':
server = socketserver.ThreadingTCPServer(server_addr, MySock)
server.serve_forever()
网络编程之 TCP-UDP的详细介绍的更多相关文章
- 4.Java网络编程之TCP/UDP
常见传输协议: UDP , TCP UDP协议: 特点: 1.将数据及源和目的封装成数据包中,不需要建立连接 2.每个数据包的大小限制在64K内 ...
- 网络编程之TCP/UDP及其流程比较(转)
TCP与UDP的区别 基于连接与无连接 对系统资源的要求(TCP较多,UDP少) UDP程序结构较简单 流模式与数据报模式TCP保证数据正确性,UDP可能丢包TCP保证数据顺序,UDP不保证 具体编程 ...
- java网络编程之TCP通讯
java中的网络编程之TCP协议的详细介绍,以及如何使用,同时我在下面举2例说明如何搭配IO流进行操作, /* *TCP *建立连接,形成传输数据的通道: *在连接中进行大数据量传输: *通过三次握手 ...
- 网络编程之TCP编程
网络编程之TCP编程 前面已经介绍过关于TCP协议的东西,这里不做赘述.Java对于基于TCP协议的网络通信提供了良好的封装,Java使用socket对象来代表两端的通信窗口,并通过Socket产生I ...
- Java网络编程之TCP、UDP
Java网络编程之TCP.UDP 2014-11-25 15:23 513人阅读 评论(0) 收藏 举报 分类: java基础及多线程(28) 版权声明:本文为博主原创文章,未经博主允许不得转载. ...
- [网络编程之Socket套接字介绍,套接字工作流程,基于TCP协议的套接字程序]
[网络编程之Socket套接字介绍,套接字工作流程,基于TCP协议的套接字程序] 为何学习socket套接字一定要先学习互联网协议: 1.首先:要想开发一款自己的C/S架构软件,就必须掌握socket ...
- 网络编程之TCP/IP各层详解
网络编程之TCP/IP各层详解 我们将应用层,表示层,会话层并作应用层,从TCP/IP五层协议的角度来阐述每层的由来与功能,搞清楚了每层的主要协议,就理解了整个物联网通信的原理. 首先,用户感知到的只 ...
- Python网络编程之TCP套接字简单用法示例
Python网络编程之TCP套接字简单用法示例 本文实例讲述了Python网络编程之TCP套接字简单用法.分享给大家供大家参考,具体如下: 上学期学的计算机网络,因为之前还未学习python,而jav ...
- Java网络编程之TCP
Java网络编程之TCP TCP主要需要两个类:Socket和ServerSocket,Socket是客户端连接服务器时创建,参数需要指定服务器的ip和端口,ServerSocket是服务器端创建 ...
- 应聘复习基础笔记1:网络编程之TCP与UDP的优缺点,TCP三次握手、四次挥手、传输窗口控制、存在问题
重要性:必考 一.TCP与UDP的优缺点 ①TCP---传输控制协议,提供的是面向连接.可靠的字节流服务.当客户和服务器彼此交换数据前,必须先在双方之间建立一个TCP连接,之后才能传输数据.TCP提供 ...
随机推荐
- CSS实现进度条
进度条经常运用于网页,即使我们意识到不是所有的东西都将瞬间被加载完成,这些进度条用于提醒使用者关于网页上具体的任务进程,譬如上传,下载,加载应用程序等. 以前如果想要创建一个进度条的动画效果,没有使用 ...
- Mybatis中#{}与${}的使用
含义 #{}:为占位符 ${}:为拼接符 区别: 用法 #{}:为参数占位符?,即sql预编译 ${}为字符串替换, 即字符串拼接 执行流程 #{}:动态解析 --> 预编译 - ...
- 微信小程序の小程序事件流
一.什么是事件? 事件是视图层到逻辑层的通讯方式:事件可以将用户的行为,反馈到逻辑层进行处理:事件可以绑定在组件上,触发事件后,就会执行逻辑层中对应的事件处理函数:事件对象可以携带额外信息. 二.事件 ...
- xfce4之whisker不显示自定义启动器的解决
对某些启动比较麻烦的程序,想创建个启动器显示在whisker里,这样就能快速启动了. 通常自己创建的desktop文件可以放~/.local/share/applications里,但是按下面这个创建 ...
- IntelliJ IDEA 添加本地xsd文件
地址: http://code.alibabatech.com/schema/dubbo/dubbo.xsd
- LOJ6485 LJJ 学二项式定理 解题报告
LJJ 学二项式定理 题意 \(T\)组数据,每组给定\(n,s,a_0,a_1,a_2,a_3\),求 \[ \sum_{i=0}^n \binom{n}{i}s^ia_{i\bmod 4} \] ...
- 状态压缩dp增量统计贡献——cf1238E(好题)
这题的状态设计非常巧妙,因为dp[S]表示的并非当前正确的值,而是维护一个中间量,这个中间量在到达末状态时才正确 当然官方的题解其实更加直观,只不过理解起来其实有点困难 /* 给定一个串s,字符集为2 ...
- PHP 图形验证码
一段生成图形验证码的代码,向原创作者致谢. 1.将以下代码保存为 txm.php ,注:直接运行该页面是没有结果的,要用另一页面引用,请看步骤2 <?php session_start(); $ ...
- flutter 使用keyboard_actions 关闭ios键盘
项目中登录 输入账号密码 弹出的键盘 关闭不了,从而 引来一些问题, 1,第一次关闭 项目是在 最外层包裹一层,点击的时候进行关闭, return Scaffold( resizeToAvoidBot ...
- angularjs 中使用 service 在controller 之间 share 对象和数据
在做angularjs 的UI 时,我们经常会遇到一个页面之间有几个controller,在controller 之间share 公共的一些数据和方法就变得比较困难,目前推荐的做法是创建一个servi ...