一。文件上传

  对于一些比较大的文件,当传输的数据大于内存时,显然,一次性将数据读取到内存中,在从内存传输到服务器显然时不可取的。

  所以,在上传文件时,可以在with open打开文件,边读取文件边发送,一行行的发送,在接收端也可以一行行的写入,这样在内存中占用的内存就只是一行而已。

  注意,在读取文件所时可以将文件已二进制的方法读取,这样就可以免去编码过程。

  如下:

import socket
import os
import json
import struct server = socket.socket()
server.bind(('127.0.0.1',8080))
server.listen(5) while True:
conn,addr = server.accept()
while True:
try:
header_len = conn.recv(4)
# 解析字典报头
header_len = struct.unpack('i',header_len)[0]
# 再接收字典数据
header_dic = conn.recv(header_len)
real_dic = json.loads(header_dic.decode('utf-8'))
# 获取数据长度
total_size = real_dic.get('file_size')
# 循环接收并写入文件
recv_size = 0
with open(real_dic.get('file_name'),'wb') as f:
while recv_size < total_size:
data = conn.recv(1024)
f.write(data)
recv_size += len(data)
print('上传成功')
except ConnectionResetError as e:
print(e)
break
conn.close()

服务器

import socket
import json
import os
import struct client = socket.socket()
client.connect(('127.0.0.1',8080)) while True:
# 获取电影列表 循环展示
MOVIE_DIR = r'D:\python脱产10期视频\day25\视频'
movie_list = os.listdir(MOVIE_DIR)
# print(movie_list)
for i,movie in enumerate(movie_list,1):
print(i,movie)
# 用户选择
choice = input('please choice movie to upload>>>:')
# 判断是否是数字
if choice.isdigit():
# 将字符串数字转为int
choice = int(choice) - 1
# 判断用户选择在不在列表范围内
if choice in range(0,len(movie_list)):
# 获取到用户想上传的文件路径
path = movie_list[choice]
# 拼接文件的绝对路径
file_path = os.path.join(MOVIE_DIR,path)
# 获取文件大小
file_size = os.path.getsize(file_path)
# 定义一个字典
res_d = {
'file_name':'性感荷官在线发牌.mp4',
'file_size':file_size,
'msg':'注意身体,多喝营养快线'
}
# 序列化字典
json_d = json.dumps(res_d)
json_bytes = json_d.encode('utf-8') # 1.先制作字典格式的报头
header = struct.pack('i',len(json_bytes))
# 2.发送字典的报头
client.send(header)
# 3.再发字典
client.send(json_bytes)
# 4.再发文件数据(打开文件循环发送)
with open(file_path,'rb') as f:
for line in f:
client.send(line)
else:
print('not in range')
else:
print('must be a number')

客户端

  需要注意的时struct在解包的时候需要使用列表【0】对其取值。

二。异常处理

  在程序中难免会出现错误,有些错误是可以解决的,比如语法错误,这种错误编译器都是会提示的,而逻辑错误有时候不可避免,因为程序员不知道其什么时侯会发生错误。

  针对这种错误,可以使用异常处理机制对其进行捕获

  常见的错误类型有:

  NAMERROR 名字错误

  SyntaxError 语法错误

  KeyError 键不存在

  ValueError 值错误

  IndexError 索引错误

  处理这些bug时,只需要将可能出些这些bug的代码放入try。。except中。try中的代码越少越号。语法如下:

try:
#可能出现错误的代码
except 错误类型 as e: #将错误信息赋值给e
#出错之后的应对措施

  在except中可以使用print(e)输出错误报告。

try:
#可能出现错误的代码
except 错误类型 as e: #将错误信息赋值给e
#出错之后的应对措施
else:
#没有出错执行的代码

  当程序中有else时,程序没有出现错误,就会执行else。

try:
#可能出现错误的代码
except 错误类型 as e: #将错误信息赋值给e
#出错之后的应对措施
else:
#没有出错执行的代码
finall:
#程序结束后执行的代码。

  当程序中有finally时,无论代码有没有异常,都会执行finally。

  操作场景,当对文件操作时,文件报错会使得文件没有关闭,占用资源,使用这个异常处理后,即使文件报错,也会把文件操作关E闭。

  raise TypeError主动抛出异常。

  当程序员需要主动抛出异常时,可以使用raise TypeError(‘’)主动抛出异常。

  assert 断言

a=8
b=9
assert a<b

  使用assert时,当程序正确时,会直接运行程序,不影响代码执行,如果错误则会抛出错误。

  自定义异常:

class MyError(BaseException):
def __init__(self,msg):
super().__init__()
self.msg=msg
def __str__(self):
return '<dfsdf%ssdfsdaf>' %self.msg raise MyError('我自己定义的异常')

  自定义异常可以定义被raise调用。出的报错信息就是__str__中的格式化信息。

三。UDP的使用

  对于tcp的服务器使用,需要先定义一个socket,使用bind设置服务器地址,调用listen监听,在阻塞等待链接。

  但是对于UDP的服务器来说没有链接池这个概念。

  由于没有双向通道,所以它也不需要阻塞等待。

import socket

server = socket.socket(type=socket.SOCK_DGRAM)  # UDP协议
server.bind(('127.0.0.1',8080))
# UDP不需要设置半连接池 它也没有半连接池的概念 # 因为没有双向通道 不需要accept 直接就是通信循环
while True:
data, addr = server.recvfrom(1024)
print('数据:',data) # 客户端发来的消息
print('地址:',addr) # 客户端的地址
server.sendto(data.upper(),addr)

  TCP的服务器,需要定义socket,并连接connet,到服务器,形成双向通道。

  而UDP则不需要connect,只需要设置发送地址。

import socket

client = socket.socket(type=socket.SOCK_DGRAM)
# 不需要建立连接 直接进入通信循环
server_address = ('127.0.0.1',8080)
while True:
client.sendto(b'hello',server_address)
data, addr = client.recvfrom(1024)
print('服务端发来的数据',data)
print('服务端的地址',addr)

  当需要发送时,使用sendto方法,对内容加地址进行发送,第二参数放地址。

  当需要接受数据时,使用recvform方法接受数据,其产生的返回值有两个,一个是接受的数据,另一个是发送数据的地址。

  udp满足以下特点:

  1.udp协议客户端允许发空

    因为udp自带包头。即使发送的是空也会发送包头。

  2.udp协议不会粘包

  3.udp协议服务端不存在的情况下,客户端照样不会报错

    因为不存在连接,是一种发短信的方式。

  4.udp协议支持并发

  udp的并发可以使一个服务器同时连接多个客户端,其中使用轮流对其返回和接受值的方法连接多个客户端。例子如下:

import socket

server = socket.socket(type=socket.SOCK_DGRAM)
server.bind(('127.0.0.1',8080)) while True:
data, addr = server.recvfrom(1024)
print(data.decode('utf-8'))
msg = input('>>>:')
server.sendto(msg.encode('utf-8'),addr)

服务端

import socket

client = socket.socket(type=socket.SOCK_DGRAM)
server_address = ('127.0.0.1',8080) while True:
msg = input('>>>:')
msg = '来自客户端5的消息:%s'%msg
client.sendto(msg.encode('utf-8'),server_address)
data, server_addr = client.recvfrom(1024)
print(data.decode('utf-8'))

客户端

  并发:看起来像同时操作

  并行:真的使同时操作

四。socketserver

  在TCP中,如果对一个服务器同时连接多个客户端,会将其连接请求置于连接池里等待,只有当一个连接断开后,才会连接下一个连接请求。

  所以,手动实现像udp一样的并发是非常困难的,但是可以使用socketserver,灵活的创建连接,使得看起来像同时运行。

  tcp的使用:

import socketserver

class MyServer(socketserver.BaseRequestHandler):
def handle(self):
# print('来啦 老弟')
while True:
data = self.request.recv(1024)
print(self.client_address) # 客户端地址
print(data.decode('utf-8'))
self.request.send(data.upper()) if __name__ == '__main__':
"""只要有客户端连接 会自动交给自定义类中的handle方法去处理"""
server = socketserver.ThreadingTCPServer(('127.0.0.1',8080),MyServer) # 创建一个基于TCP的对象
server.serve_forever() # 启动该服务对象

服务器

import socket

client = socket.socket()
client.connect(('127.0.0.1',8080)) while True:
client.send(b'hello')
data = client.recv(1024)
print(data.decode('utf-8'))

客户端

  当调用一个ThreadingTCPServer方法时,需要传入两个参数,一个时服务器的地址,另一个是一个类,类中继承了socketserver中hander的方法,方法中写入对该客户端的操作,发送的数据等。

  客户端和一般一样。

  udp的使用。

  对于UDP,socketserver也提供了方法去操作,所调用的方法是ThreadingUDPServer。

import socketserver

class MyServer(socketserver.BaseRequestHandler):
def handle(self):
# print('来啦 老弟')
while True:
data,sock = self.request
print(self.client_address) # 客户端地址
print(data.decode('utf-8'))
sock.sendto(data.upper(),self.client_address) if __name__ == '__main__':
"""只要有客户端连接 会自动交给自定义类中的handle方法去处理"""
server = socketserver.ThreadingUDPServer(('127.0.0.1',8080),MyServer) # 创建一个基于TCP的对象
server.serve_forever() # 启动该服务对象

服务器

import socket
import time client = socket.socket(type=socket.SOCK_DGRAM)
server_address = ('127.0.0.1',8080) while True:
client.sendto(b'hello',server_address)
data,addr = client.recvfrom(1024)
print(data.decode('utf-8'),addr)
time.sleep(1)

客户端

  其中,request返回的也是两个值,一个是返回的数据,另一个是可以对其调用的sock

day29 8_8 TCP上传文件socketserver的应用的更多相关文章

  1. java 利用TCP上传文件

    从客户端上传到服务器端,其实本质上也就是复制! package july76net; //上传文件(文本) import java.io.BufferedReader; import java.io. ...

  2. java 26 - 9 网络编程之 TCP协议多用户上传文件

    TCP实现多用户上传文件: 需要同时给多用户上传文件,这样就得用多线程来实现. 实际上,这样的话,上传的先后顺序和速度就跟客户端的带宽有关:带宽够,就容易抢占到线程的执行权: 首先,创建个线程类:(这 ...

  3. 那些年的 网络通信之 TCP/IP 传输控制协议 ip 加 端口 客户端上传文件到服务器端服务器端返回上传成功消息

    多线程开启, 客户端通过 Socket 流 上传文件到服务端的一个小程序练习. 1. 抓住阻塞式方法,去调试 2. 获取对应流对象操作对应的对象 这时候自己不能懵,一定要清晰,最好命名就能区别,一搞混 ...

  4. 通过HTTP协议上传文件

         HTTP是很常见的协议,虽然用得很多,但对细节的了解却是很浅,这回通过向服务端上传文件信息来理解细节.网络库的选择:1.WinHTTP是windows下常用的库:2.CURL是广受喜爱的开源 ...

  5. 使用Servlet上传文件

    使用浏览器向服务器上传文件其本质是打开了一个长连接并通过TCP方式传输数据.而需要的动作是客户端在表单中使用file域,并指定该file域的name值,然后在form中设定enctype的值为mult ...

  6. 记一次FTP上传文件总是超时的解决过程

    好久没写博,还是重拾记录一下吧. 背景:买了一个阿里云的云虚拟机用来搭建网站(起初不了解云虚拟主机和云服务器的区别,以为都是有SSH功能的,后来发现不是这样样子啊,云虚拟机就是FTP上传网页+MySQ ...

  7. C# 通过WebService方式 IIS发布网站 上传文件到服务器

    应用场景:要将本地的文件 上传到服务器的虚拟机上 网络环境:公司局域网(如下图中第二种) 开发环境:VS2010 服务器环境:WinServer2008    虚拟机环境:WinServer2008 ...

  8. NSURLSession/NSURLConnection的上传文件方法(已做了更新)

    最好的学习方法就是 领悟 + 证悟. 此篇文章的理论基础主要是与HTTP网络通信协议相关.为集中精力,可以先把TCP/IP协议这些置之不理,也就是先只关注HTTP的请求和响应的结构.HTTP完整的原理 ...

  9. 三种方式上传文件-Java

    前言:负责,因为该项目他(jetty嵌入式开始SpringMvc)实现文件上传的必要性,并拥有java文件上传这一块还没有被曝光.并 Http 更多晦涩协议.因此,这种渐进的方式来学习和实践上载文件的 ...

随机推荐

  1. Mybatis日志(七)

    Mybatis内置的日志工厂提供日志功能,具体的日志实现有以下几种工具: SLF4J Apache Commons Logging Log4j 2 Log4j JDK logging 具体选择哪个日志 ...

  2. 洛谷 P4316 绿豆蛙的归宿

    洛谷 P4316 绿豆蛙的归宿 洛谷传送门 题目背景 随着新版百度空间的上线,Blog宠物绿豆蛙完成了它的使命,去寻找它新的归宿. 题目描述 给出一个有向无环图,起点为1终点为N,每条边都有一个长度, ...

  3. SQL Server 修改数据库

    1. 可视化界面修改数据库 (1)右击数据库,然后选择属性. (2)在工具选项卡中,选择[文件]页,可以更改所有者,文件大小,自增量等参数. 2.  使用ALTER Database修改数据库 (1) ...

  4. -bash:vi:command not find 问题解决

    Linux命令行输入命令执行后报“bash:vi:command not found”. 这是由于系统PATH设置问题,PATH没有设置对,系统就无法找到精确命令了. 1.在命令行中输入:export ...

  5. 1130不允许连接到MySql server

    连接远程服务器mysql时,报错: 1130-host ... is not allowed to connect to this MySql server 这是因为默认只让localhost的主机连 ...

  6. USACO19JAN Gold题解

    噩梦的回忆.. 上周日在机房打的模拟赛,结果十分惨烈,就最后一题yy出了正解结果玄学的只拿了80 考试结果:0+0+80=80 订正时对着T3打了2hours结果还是90 订正结果:100+100+9 ...

  7. yum安装软件报错Error: Nothing to do

    今天在一台新服务器上装一些常用软件,一开始安装ncdu(一个很好用的磁盘分析工具,用来查找大文件),报错如下: 在网上找了各种办法,什么更新yum啊,清理yum缓存啊的,统统没用 最后的找到的问题是, ...

  8. java8 LinkedHashMap 原理

    LinkedHashMap 原理 基于jdk1.8 HashMap原理:http://www.cnblogs.com/zhaojj/p/7805376.html LinkedHashMap 继承Has ...

  9. Linux下启动SpringBoot打包的jar

    前言 这两天把视力档案后台部署的方式改了一下,由原来打包成war包,部署到一个tomcat里面,转变成直接打包成jar包,然后使用 java -jar命令进行启动 下面讲讲遇到的问题 1)java - ...

  10. ubuntu python 版本管理

    ubuntu 命令行查看 python 目录 $ whereis python # 显示所有得到 python 目录 $ which python  # 显示默认的 python 解释器目录 $ wh ...