脚本之家,脚本语言编程技术及教程分享平台!
分类导航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|shell|

服务器之家 - 脚本之家 - Python - Python Flask实现图片上传与下载的示例详解

Python Flask实现图片上传与下载的示例详解

2023-02-23 11:30beautifulzzzz Python

这篇文章主要为大家详细介绍了如何利用Python和Flask实现图片上传与下载(支持漂亮的拖拽上传),文中示例代码讲解详细,感兴趣的可以了解一下

1、效果预览

我们基于 Flask 官方指导工程,增加一个图片拖拽上传功能,效果如下:

Python Flask实现图片上传与下载的示例详解

 

2、新增逻辑概览

我们在官方指导工程上进行增加代码,改动如下:

Python Flask实现图片上传与下载的示例详解

由于 flask 官方 Demo 基于蓝图设计,这给我们新增逻辑带来了很大的方便。关于官方 Demo 的介绍,可以参考《Flask 入门(以一个博客后台为例)》

 

3、tuchuang.py 逻辑介绍

3.1 图片上传

1)该接口采用 POST 方法,需要登录;

2)接着,检查请求中是否有 'file' 关键词,然后取出文件,判断文件是否为空或是否合法;

3)最后,将上传的图片保存(采用秒级别的时间戳+随机数重命名);

4)该接口在上传图片成功后,返回该图片的链接;如果不成功,返回 upload.html 页面;

@bp.route('/', methods=['GET', 'POST'])
@login_required
def upload_file():
  if request.method == 'POST':
      # check if the post request has the file part
      if 'file' not in request.files:
          flash('No file part')
          return redirect(request.url)
      file = request.files['file']
      # If the user does not select a file, the browser submits an
      # empty file without a filename.
      if file.filename == '':
          flash('No selected file')
          return redirect(request.url)
      if file and allowed_file(file.filename):
          # 获取安全的文件名 正常文件名
          filename = secure_filename(file.filename)
          
          # 生成随机数
          random_num = random.randint(0, 100)
          # f.filename.rsplit('.', 1)[1] 获取文件的后缀
          filename = datetime.now().strftime("%Y%m%d%H%M%S") + "_" + str(random_num) + "." + filename.rsplit('.', 1)[1]
          file_path = app.config['UPLOAD_FOLDER']    # basedir 代表获取当前位置的绝对路径
          
          # 如果文件夹不存在,就创建文件夹
          if not os.path.exists(file_path):
	            os.makedirs(file_path)
	
          file.save(os.path.join(file_path, filename))
          return redirect(url_for('tuchuang.download_file', name=filename))
  return render_template("tuchuang/upload.html")

3.2 图片合法检查

上述代码中有一个合法检测的函数allowed_file,用于检查上传图片的后缀是否在允许列表:

basedir = os.path.abspath(os.path.dirname(__file__))                 # 获取当前文件所在目录
UPLOAD_FOLDER = basedir+'/static/file/img'                           # 计算图片文件存放目录
ALLOWED_EXTENSIONS = {'txt', 'pdf', 'png', 'jpg', 'jpeg', 'gif'}     # 设置可上传图片后缀 

app = Flask(__name__)
app.config['UPLOAD_FOLDER'] = UPLOAD_FOLDER
bp = Blueprint("tuchuang", __name__, url_prefix="/tuchuang")

def allowed_file(filename):                                          # 检查上传图片是否在可上传图片允许列表
  return '.' in filename and \
         filename.rsplit('.', 1)[1].lower() in ALLOWED_EXTENSIONS

3.3 图片下载

图片下载比较简单,就是调用send_from_directory函数,就能够把 static 目录下的对应文件发出:(我们一般把各种用于外面访问的静态图片、JS、CSS 等放在 static 文件中)

@bp.route('/download/<name>')
def download_file(name):
  return send_from_directory(app.config["UPLOAD_FOLDER"], name)

 

4、__init__.py 逻辑介绍

由于我们采用蓝图设计,因此需要稍微修改下__init__.py文件,来将 tuchuang.py 加入:

Python Flask实现图片上传与下载的示例详解

  • MAX_CONTENT_LENGTH=16 * 1000 * 1000上传图片大小限制
  • from flaskr import auth, blog, tuchuang
  • app.register_blueprint(tuchuang.bp)将 tuchuang 加入蓝图
  • app.add_url_rule("/download/<name>", endpoint="download_file", build_only=True)

 

5、upload.html 介绍

5.1 upload Jinja 模板介绍

  • Jinja 引用外部 css:<link rel="stylesheet" href="{{ url_for('static', filename='file/css/upload.css') }}">
  • Jinja 引用外部 js:<script type="text/javascript" src="{{ url_for('static', filename='file/js/upload.js') }}"></script>
  • 该 Jinja 模板实现了两种图片上传交互:

普通版,采用 file select 框 + submit 按钮,实现图片上传:

<form method=post enctype=multipart/form-data>
  <input type=file name=file>
  <input type=submit value=Upload>
</form>

拖拽版(需要借助 JS,CSS),在<div id="drop-area">内实现

下面是tuchuang/upload.html完整代码:

<!doctype html>
<link rel="stylesheet" href="{{ url_for('static', filename='file/css/upload.css') }}" rel="external nofollow" >
<script type="text/javascript"  src="{{ url_for('static', filename='file/js/upload.js') }}"></script>
<title>Upload new File</title>
<h1>Upload new File</h1>
<form method=post enctype=multipart/form-data>
  <input type=file name=file>
  <input type=submit value=Upload>
</form>
<div id="drop-area">
  <form class="my-form">
      <p>Upload multiple files with the file dialog or by dragging and dropping images onto the dashed region</p>
      <input type="file" id="fileElem" multiple accept="image/*" onchange="handleFiles(this.files)">
      <label class="button" for="fileElem">Select some files</label>
      <div id="gallery"></div>
      <progress id="progress-bar" max=100 value=0></progress>
  </form>
</div>

5.2 upload css 介绍(虚线框)

下面是拖拽需要用到的 CSS,大家暂时浏览下,之后结合 JS 就明白了:

#drop-area {
  border: 2px dashed #ccc;
  border-radius: 20px;
  width: 480px;
  font-family: sans-serif;
  margin: 100px auto;
  padding: 20px;
}
#drop-area.highlight {
  border-color: purple;
}
p {
  margin-top: 0;
}
.my-form {
  margin-bottom: 10px;
}
#gallery {
  margin-top: 10px;
}
#gallery img {
  width: 150px;
  margin-bottom: 10px;
  margin-right: 10px;
  vertical-align: middle;
}
.button {
  display: inline-block;
  padding: 10px;
  background: #ccc;
  cursor: pointer;
  border-radius: 5px;
  border: 1px solid #ccc;
}
.button:hover {
  background: #ddd;
}
#fileElem {
  display: none;
}

5.3 upload js 介绍(拖拽)

5.3.1 JS 拖拽框架

JS 代码主要基于 window.onload + 拖拽事件实现,大致框架如下:

window.onload=function(){
  var dropArea = document.getElementById('drop-area')

  // 阻止默认行为
  ;['dragenter', 'dragover', 'dragleave', 'drop'].forEach(eventName => {
      dropArea.addEventListener(eventName, preventDefaults, false)
  })

  function preventDefaults (e) {
      e.preventDefault()
      e.stopPropagation()
  }

  // 增加事件,鼠标拖入边框高亮,拖出边框变为原来样子
  ;['dragenter', 'dragover'].forEach(eventName => {
      dropArea.addEventListener(eventName, highlight, false)
  })

  ;['dragleave', 'drop'].forEach(eventName => {
      dropArea.addEventListener(eventName, unhighlight, false)
  })

  function highlight(e) {
      dropArea.classList.add('highlight')
  }

  function unhighlight(e) {
      dropArea.classList.remove('highlight')
  }

  // 增加事件,鼠标放下,之后准备上传图片
  dropArea.addEventListener('drop', handleDrop, false)

  function handleDrop(e) {
  	// 之后准备上传图片
  }
}

window.onload() 方法用于在网页加载完毕后立刻执行的操作,即当 HTML 文档加载完毕后,立刻执行某个方法。

为什么使用 window.onload()?

因为 JavaScript 中的函数方法需要在 HTML 文档渲染完成后才可以使用,如果没有渲染完成,此时的 DOM 树是不完整的,这样在调用一些 JavaScript 代码时就可能报出"undefined"错误。

5.3.2 JS 图片上传

function handleDrop(e) {
	// 从拖拽放下事件中获取拖拽的文件
  let dt = e.dataTransfer
  let files = dt.files

	// 调用图片处理函数,对图片进行处理
  handleFiles(files)
}

function handleFiles(files) {
	// 对于多个图片,循环调用 uploadFile 函数,进行上传
  ([...files]).forEach(uploadFile)
}

function uploadFile(file) {
	// JS 合成表单,利用 POST 方法,实现上传(部署在远端时,要改下下面的 url)
  let url = 'http://127.0.0.1:5000/tuchuang/'
  let formData = new FormData()

  formData.append('file', file)

  fetch(url, {
      method: 'POST',
      body: formData
  })
      .then(progressDone) // <- Add `progressDone` call here
      .catch(() => { /* Error. Inform the user */ })
}

Fetch API 提供了一个 JavaScript接口,用于访问和操纵HTTP管道的部分,例如请求和响应。它还提供了一个全局 fetch()方法,该方法提供了一种简单,合理的方式来跨网络异步获取资源。详细介绍参考JavaScript使用Fetch的方法详解

  • 1.进行 fetch 请求 参考;
  • 2.支持的请求参数参考;
  • 3.发送带凭据的请求参考;
  • 4.上传 JSON 数据参考;
  • 5.上传文件参考;
  • 6.上传多个文件参考;
  • 7.检测请求是否成功参考;
  • 8.自定义请求对象参考;
  • 9.Headers参考;
  • 10.Guard参考;
  • 11.Response 对象参考;
  • 12.Body参考;
  • 13.特性检测参考;

该文章讲的比较好,大家可以跳转过去学习下~

5.3.3 JS 图片上传进度条

想要带有进度条,我们需要修改下 handleFiles 函数:

var filesDone = 0
var filesToDo = 0
var progressBar = document.getElementById('progress-bar')

...

// 预览
function previewFile(file) {
  let reader = new FileReader()
  reader.readAsDataURL(file)
  reader.onloadend = function() {
      let img = document.createElement('img')
      img.src = reader.result
      document.getElementById('gallery').appendChild(img)
  }
}

// 进度条初始化,fileDone 置 0,filesToDo 置需要上传图片总数
function initializeProgress(numfiles) {
  progressBar.value = 0
  filesDone = 0
  filesToDo = numfiles
}

// 注意,该函作为 fetch 的返回回调函数,意思是每次传输完成一个图片,进度条进行相应变化
function progressDone() {
  filesDone++
  progressBar.value = filesDone / filesToDo * 100
}

function handleFiles(files) {
  files = [...files]
  initializeProgress(files.length) 
  files.forEach(uploadFile)
  files.forEach(previewFile)
}

到此这篇关于Python Flask实现图片上传与下载的示例详解的文章就介绍到这了,更多相关Python Flask图片上传下载内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://www.cnblogs.com/zjutlitao/p/16325464.html

延伸 · 阅读

精彩推荐
  • Python以tensorflow库为例讲解Pycharm中如何更新第三方库

    以tensorflow库为例讲解Pycharm中如何更新第三方库

    这篇文章主要介绍了以tensorflow库为例讲解Pycharm中如何更新第三方库,文章介绍有详细流程,需要的小伙伴可以参考一下,希望对你的学习工作有所帮助...

    我这一次9562022-10-26
  • Pythonpython迭代dict的key和value的方法

    python迭代dict的key和value的方法

    今天小编就为大家分享一篇python迭代dict的key和value的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    手艺人小在12922021-03-13
  • PythonPython 最大概率法进行汉语切分的方法

    Python 最大概率法进行汉语切分的方法

    今天小编就为大家分享一篇Python 最大概率法进行汉语切分的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    qijingpei9282021-04-29
  • PythonPython测试网络连通性示例【基于ping】

    Python测试网络连通性示例【基于ping】

    这篇文章主要介绍了Python测试网络连通性,结合实例形式分析了Python通过发送ping请求测试网络连通性相关操作技巧,需要的朋友可以参考下...

    Teingi11252021-03-25
  • PythonDjango Admin 实现外键过滤的方法

    Django Admin 实现外键过滤的方法

    下面小编就为大家带来一篇Django Admin 实现外键过滤的方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    ishouyong6972020-12-10
  • PythonPython实现TCP/IP协议下的端口转发及重定向示例

    Python实现TCP/IP协议下的端口转发及重定向示例

    这篇文章主要介绍了Python实现TCP/IP协议下的端口转发及重定向示例,以一个webpy站点在本机的两个端口双向通信下演示,需要的朋友可以参考下...

    Cedric Porter7872020-08-27
  • PythonPython实现以时间换空间的缓存替换算法

    Python实现以时间换空间的缓存替换算法

    缓存是指可以进行高速数据交换的存储器,它先于内存与CPU交换数据,因此速度很快。缓存就是把一些数据暂时存放于某些地方,可能是内存,也有可能硬...

    Python教程网5582020-08-13
  • PythonPython 中拼音库 PyPinyin 用法详解

    Python 中拼音库 PyPinyin 用法详解

    很多朋友问小编怎样把一批中文文件转拼音命名呢?下面就让我们来了解 Python 的一个库 PyPinyin 吧,感兴趣的朋友跟随小编一起看看吧...

    程序员阿城4952021-11-16