热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

网络编程基础二

[TOC]subprocess模块?作用:?1、可以通过代码执行操作系统的终端命令?2、并返回终端执行命令后的结果TCP协议特性:是一个流式协议,会将多次连续发送的数据量小、间隔时


目录



  • subprocess模块

  • TCP协议

  • 黏包问题

  • 上传大文件

  • UDP

  • 简易聊天室

  • SocketServer




subprocess模块

? 作用:

? 1、可以通过代码执行操作系统的终端命令

? 2、并返回终端执行命令后的结果

用法:
import subprocess

cmd = inpu('CMD >>>')
obj = subprocess.Popen(
cmd.decode('utf8'),
shell=True,
stdout= subprocess.PIPE,
stderr= subprocess.PIPE
)
result = obj.stdout.read() + obj.stderr.read()

print(result.decode('gbk'))

TCP协议

特性:是一个流式协议,会将多次连续发送的数据量小、间隔时间短的数据一次性打包发送

黏包问题

? 原因:

? 1、无法预测需要接收的数据的长度

? 2、多次连续发送数据量小、并且时间间隔短的数据一次性打包发送。

解决黏包问题: 告诉对端数据的确切长度。

-struct模块:

? 必须先发送报头,再发送数据:

? 1、自定义报头,将数据长度等数据的描述信息制作成字典,

? 2、将字典json.dumps序列化并编码转为bytes,用于传输

? 3、先将报头的长度,用struct.pack格式化成固定长度(4字节)

? 4、发送:先发送报头长度,再发送报头,然后发送数据正文

? 接收:

? 1、先接收到4个字节的报头长度信息,unpack解压得到报头的长度值

? 2、根据报头长度值接收报头,解码并json.load反序列得到数据正文的描述信息(包含数据长度)

? 3、根据数据长度接收数据正文,

? 用法:

import json,struct
#假设通过客户端上传10G:10000000的文件:小视频.txt
#为避免粘包,必须自定制报头
header={'file_size':10000000,'file_name':'小视频.txt', 'md5':'8f6fbf8347faa4924a76856701edb0f3'} #10G数据,文件路径和md5值
#为了该报头能传送,需要序列化并且转为bytes
head_bytes=bytes(json.dumps(header),encoding='utf-8') #序列化并转成bytes,用于传输
#为了让客户端知道报头的长度,用struck将报头长度这个数字转成固定长度:4个字节
head_len_bytes=struct.pack('i',len(head_bytes)) #这4个字节里只包含了一个数字,该数字是报头的长度
#客户端开始发送
conn.send(head_len_bytes) #先发报头的长度,4个bytes
conn.send(head_bytes) #再发报头的字节格式
conn.sendall(文件内容) #然后发真实内容的字节格式
#服务端开始接收
head_len_bytes=s.recv(4) #先收报头4个bytes,得到报头长度的字节格式
x=struct.unpack('i',head_len_bytes)[0] #提取报头的长度
head_bytes=s.recv(x) #按照报头长度x,收取报头的bytes格式
header=json.loads(json.dumps(header)) #提取报头
#最后根据报头的内容提取真实的数据,比如
real_data_len=s.recv(header['file_size'])
s.recv(real_data_len)

上传大文件

以上传视频为例:

客户端:

import socket
import struct
import json
client = socket.socket()
client.connect(
('127.0.0.1', 9527)
)
with open(r'C:\day 027\5 上传大文件.mp4', 'rb')as f:
movie_bytes = f.read()
head_dic = {
'file_name': '5 上传大文件.mp4',
'file_size': len(movie_bytes)
}
json_head = json.dumps(head_dic)
bytes_head = json_head.encode('utf-8')
headers = struct.pack('i', len(bytes_head))
client.send(headers)
client.send(bytes_head)
init_data = 0
num = 1
with open(r'C:\day 027\5 上传大文件.mp4', 'rb')as f:
while init_data send_data = f.read(1024)
num += 1
client.send(send_data)
print(send_data, f'第{num}段数据发送完成')
init_data += len(send_data)
print('发送完毕!')

服务端:

import socket
import json
import struct
server = socket.socket()
server.bind(
('127.0.0.1', 9527)
)
server.listen(5)
while True:
conn, addr = server.accept()
try:
headers = conn.recv(4)
head_len = struct.unpack('i', headers)[0]
byte_head = conn.recv(head_len)
head_dic = json.loads(byte_head.decode('utf-8'))
print(head_dic)
file_name = head_dic.get('file_name')
file_size = head_dic.get('file_size')
init_data = 0
num = 1
with open(file_name, 'wb') as f:
while init_data data = conn.recv(1024)
print(f'第{num}段',data)
f.write(data)
init_data += len(data)
print(f'{file_name}接收完毕!')
except Exception as e:
print('error:', e)
break
conn.close()

UDP

? UDP是一种传输协议

? 1、不需要建立双向管道

? 2、不会黏包

? 3、客户端给服务端发送数据,不需要等服务端的回执

? 4、服务端只接收需要接收的数据,不管客户端传过来多少

服务端:

import socket
server = socket.socket(type=socket.SOCK_DGRAM)
server.bind(
('127.0.0.1', 9527)
)
msg1 = server.recvfrom(1024)
msg2 = server.recvfrom(1024)
msg3 = server.recvfrom(1024)
print(msg1, msg2, msg3)

客户端:

import socket
client =socket.socket(type=socket.SOCK_DGRAM)
server_ip_port = ('127.0.0.1', 9527)
client.sendto(b'hello!', server_ip_port)
client.sendto(b'hello!', server_ip_port)
client.sendto(b'hello!', server_ip_port)
client.sendto(b'hello!', server_ip_port)
client.sendto(b'hello!', server_ip_port)

执行结果:(b‘hello!‘, (‘127.0.0.1‘, 57182)) (b‘hello!‘, (‘127.0.0.1‘, 57182)) (b‘hello!‘, (‘127.0.0.1‘, 57182))

简易聊天室

基于TCP协议,用socket套接字实现一个服务端对多个客户端的简易聊天室:

服务端:

import socket
server = socket.socket(type=socket.SOCK_DGRAM)
server.bind(
('127.0.0.1', 9527)
)
while True:
# 服务端接收客户端传过来的消息
msg, addr = server.recvfrom(1024) # (消息,客户端地址)
msg1, addr1 = server.recvfrom(1024) # (消息,客户端地址)
msg2, addr2 = server.recvfrom(1024) # (消息,客户端地址)
print(addr)
print(addr1)
print(addr2)
print(msg.decode('utf-8'))
print(msg1.decode('utf-8'))
print(msg2.decode('utf-8'))
# 服务端往客户端发送消息
send_msg = input('服务端发送消息:').encode('utf-8')
server.sendto(send_msg, addr)
server.sendto(send_msg, addr1)
server.sendto(send_msg, addr2)

客户端1:

import socket
client = socket.socket(type=socket.SOCK_DGRAM)
server_ip_port = ('127.0.0.1', 9527)
while True:
send_msg = input('客户端1: ').encode('utf-8')
# 发送消息必须要加上对方地址
client.sendto(send_msg, server_ip_port)
# 能接收任何人的消息
msg = client.recv(1024)
print(msg.decode('utf-8'))

客户端2:

import socket
client = socket.socket(type=socket.SOCK_DGRAM)
server_ip_port = ('127.0.0.1', 9527)
while True:
send_msg = input('客户端2: ').encode('utf-8')
# 发送消息必须要加上对方地址
client.sendto(send_msg, server_ip_port)
# 能接收任何人的消息
msg = client.recv(1024)
print(msg.decode('utf-8'))

客户端3:

import socket
client = socket.socket(type=socket.SOCK_DGRAM)
server_ip_port = ('127.0.0.1', 9527)
while True:
send_msg = input('客户端3: ').encode('utf-8')
# 发送消息必须要加上对方地址
client.sendto(send_msg, server_ip_port)
# 能接收任何人的消息
msg = client.recv(1024)
print(msg.decode('utf-8'))

先启动服务端,再依次启动客户端,可开始模拟聊天室聊天

SocketServer

? 是python内置模块,可以用来简化socket套接字服务端的代码

import socket
class MyTcpServer(socketserver, BaseRequestHandler):
def handler(self):

data = self.request.recv(1024)
pirnt(data)
send_msg = input('服务端:').encode('utf8')
self.request.send(send_msg)


推荐阅读
  • Framework7:构建跨平台移动应用的高效框架
    Framework7 是一个开源免费的框架,适用于开发混合移动应用(原生与HTML混合)或iOS&Android风格的Web应用。此外,它还可以作为原型开发工具,帮助开发者快速创建应用原型。 ... [详细]
  • 本文介绍了如何使用 CMD 批处理脚本进行文件操作,包括将指定目录下的 PHP 文件重命名为 HTML 文件,并将这些文件复制到另一个目录。 ... [详细]
  • 解决Bootstrap DataTable Ajax请求重复问题
    在最近的一个项目中,我们使用了JQuery DataTable进行数据展示,虽然使用起来非常方便,但在测试过程中发现了一个问题:当查询条件改变时,有时查询结果的数据不正确。通过FireBug调试发现,点击搜索按钮时,会发送两次Ajax请求,一次是原条件的请求,一次是新条件的请求。 ... [详细]
  • 本文详细介绍了DMA控制器如何通过映射表处理来自外设的请求,包括映射表的设计和实现方法。 ... [详细]
  • Spark中使用map或flatMap将DataSet[A]转换为DataSet[B]时Schema变为Binary的问题及解决方案
    本文探讨了在使用Spark的map或flatMap算子将一个数据集转换为另一个数据集时,遇到的Schema变为Binary的问题,并提供了详细的解决方案。 ... [详细]
  • 第二十五天接口、多态
    1.java是面向对象的语言。设计模式:接口接口类是从java里衍生出来的,不是python原生支持的主要用于继承里多继承抽象类是python原生支持的主要用于继承里的单继承但是接 ... [详细]
  • 在使用Eclipse进行调试时,如果遇到未解析的断点(unresolved breakpoint)并显示“未加载符号表,请使用‘file’命令加载目标文件以进行调试”的错误提示,这通常是因为调试器未能正确加载符号表。解决此问题的方法是通过GDB的`file`命令手动加载目标文件,以便调试器能够识别和解析断点。具体操作为在GDB命令行中输入 `(gdb) file `。这一步骤确保了调试环境能够正确访问和解析程序中的符号信息,从而实现有效的调试。 ... [详细]
  • 网站访问全流程解析
    本文详细介绍了从用户在浏览器中输入一个域名(如www.yy.com)到页面完全展示的整个过程,包括DNS解析、TCP连接、请求响应等多个步骤。 ... [详细]
  • 自定义滚动条美化页面内容
    当页面内容超出显示范围时,为了提升用户体验和页面美观,通常会添加滚动条。如果默认的浏览器滚动条无法满足设计需求,我们可以自定义一个符合要求的滚动条。本文将详细介绍自定义滚动条的实现过程。 ... [详细]
  • importpymysql#一、直接连接mysql数据库'''coonpymysql.connect(host'192.168.*.*',u ... [详细]
  • 微软推出Windows Terminal Preview v0.10
    微软近期发布了Windows Terminal Preview v0.10,用户可以在微软商店或GitHub上获取这一更新。该版本在2月份发布的v0.9基础上,新增了鼠标输入和复制Pane等功能。 ... [详细]
  • 两个条件,组合控制#if($query_string~*modviewthread&t(&extra(.*)))?$)#{#set$itid$1;#rewrite^ ... [详细]
  • 解决Win10下MySQL连接问题:Navicat 2003无法连接到本地MySQL服务器(10061)
    本文介绍如何在Windows 10环境下解决Navicat 2003无法连接到本地MySQL服务器的问题,包括启动MySQL服务和检查配置文件的方法。 ... [详细]
  • 在 LeetCode 的“有效回文串 II”问题中,给定一个非空字符串 `s`,允许删除最多一个字符。本篇深入解析了如何判断删除一个字符后,字符串是否能成为回文串,并提出了高效的优化算法。通过详细的分析和代码实现,本文提供了多种解决方案,帮助读者更好地理解和应用这一算法。 ... [详细]
  • 系统数据实体验证异常:多个实体验证失败的错误处理与分析
    在使用MVC和EF框架进行数据保存时,遇到了 `System.Data.Entity.Validation.DbEntityValidationException` 错误,表明存在一个或多个实体验证失败的情况。本文详细分析了该错误的成因,并提出了有效的处理方法,包括检查实体属性的约束条件、调试日志的使用以及优化数据验证逻辑,以确保数据的一致性和完整性。 ... [详细]
author-avatar
亦涵Doris
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有