python下载文件损坏_解决python web cgi文件下载功能的bug

解决python web cgi文件下载功能的bug,下载的文件比源文件大,二进制文件直接损坏打不开

f636afc379310a5558a4601af0637eaf8326100c.jpeg?token=7f2b09e3024c78d5cb4800954d0133ab&s=31B4ED3609466EC814D488DA020090B3python

最近我在用python cgi做web开发,写文件下载功能时,参考了菜鸟教程的代码。

代码做了一些修改,代码如下:

#!C:/Users/Administrator/AppData/Local/Programs/Python/Python37/python.exe

#第一行是python解释器的路径,绝对不能省,!也不能省略,路径根据自己安装的修改

import os

file_path='1.txt' #把文件路径'1.txt'赋值给file_path

file_name=os.path.basename(file_path) #把文件名赋值给file_name

# HTTP 头部开始

print ('Content-Disposition: attachment; filename=%s'%(file_name))

#Content-Disposition: attachment是弹出文件下载的对话框

#filename=%s等号后面是文件下载对话框显示的文件名,%s是格式化输出,%(file_name)用括号中的变量传参给%s

print () #输出1行空行,告诉浏览器HTTP头部结束

# HTTP 头部结束

fd = open(file_path, "rb") #以读取二进制的方式打开一个文件,返回的文件描述符,赋值给fd

read_bytes = fd.read() #通过文件描述符fd读取打开文件的数据,并赋值给read_bytes

print (read_bytes) #输出读取到的数据

fd.close() #关闭文件描述符fd

假如1.txt文件里的内容是:

人生苦短,我用python

那么下载以后就是这样子的:

b'\xe4\xba\xba\xe7\x94\x9f\xe8\x8b\xa6\xe7\x9f\xad\xef\xbc\x8c\xe6\x88\x91\xe7\x94\xa8python'

问:为什么会这个样子呢?

答:原因是因为print()把字节流转成字符串输出了,而且下载的文件跟源文件的大小也是不一样的,源文件大小是:27Byte(27字节),下载的文件大小是:95Byte(95字节)。

描述:如果是下载二进制文件,比如图片、视频、音乐,下载之后,文件会变大好几倍。

问:为什么会这样子呢?

答:就是因为print()的关系,所以不管源文件是什么文件,下载之后都会变成文本文件。

问:那怎么解决这个bug呢?

答:答案就是不用print()进行输出

问:有的人脑海中就会有新的问题飘出来了,不用print()怎么输出?

答:可以使用sys模块的stdout,stdout是标准输出流,只支持字节流输出,输出的内容就是这种的:b'内容',print()也是对标准输出流进行了封装。

废话不多说,看代码:

#!C:/Users/Administrator/AppData/Local/Programs/Python/Python37/python.exe

#第一行是python解释器的路径,绝对不能省,!也不能省略,路径根据自己安装的修改

import os,sys

file_path='1.txt' #把文件路径'1.txt'赋值给file_path

file_name=os.path.basename(file_path) #获取文件名,并赋值给file_name

file_size=os.path.getsize(file_path) #获取文件大小,并赋值给file_size

# HTTP 头部开始

print ('Content-Disposition: attachment; filename=%s'%(file_name))

#Content-Disposition: attachment是弹出文件下载的对话框

#filename=%s等号后面是文件下载对话框显示的文件名,%s是格式化输出,%(file_name)用括号中的变量传参给%s

print('Accept-Length:%s'%(file_size)) #这个是文件下载框显示的文件大小

print() #输出1行空行,告诉浏览器HTTP头部结束

# HTTP 头部结束

fd = open(file_path, "rb") #以读取二进制的方式打开一个文件,返回的文件描述符,赋值给fd

read_bytes = fd.read() #通过文件描述符fd读取打开文件的数据,并赋值给read_bytes

sys.stdout.flush() #刷新标准输出流的缓冲区,如果不刷新的话,客户端就接收不到数据

sys.stdout.buffer.write (read_bytes) #把读取到的数据写入到标准输出流的缓冲区

fd.close() #关闭文件描述符fd

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值