Django项目实战之用户头像上传与访问的示例


Posted in Python onApril 21, 2018

1 将文件保存到服务器本地

upload.html

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
</head>
<body>
<form action="" method="post" enctype="multipart/form-data">
  {% csrf_token %}
  <div>用户名:<input type="text" name="username"></div>
  <div>头像<input type="file" name="avatar"></div>
  <input type="submit" value="提交">
</form>
</body>
</html>

urls.py

from django.conf.urls import url
from app01 import views
urlpatterns = [
  url(r'^upload',views.upload)
]

views.py

from django.shortcuts import render,HttpResponse 
def upload(request):
  if request.method == 'POST':
    name = request.POST.get('username')
    avatar = request.FILES.get('avatar')
    with open(avatar.name,'wb') as f:
      for line in avatar:
        f.write(line)
    return HttpResponse('ok')
  return render(request,'upload.html')

总结

这样,我们就做好了一个基本的文件上传小示例,这里需要注意的有几点:
 1.form表单里需要加上csrf_token验证
 2.文件的input框的type的值为file
 3.在视图函数中获取文件要用request.FILES.get()方法
 4.通过obj.name可以获取文件的名字

2 将文件上传到数据库

models.py

from django.db import models
 class User(models.Model):
  username = models.CharField(max_length=16)
  avatar = models.FileField(upload_to='avatar')

views.py

def upload(request):
  if request.method == 'POST':
    name = request.POST.get('username')
    avatar = request.FILES.get('avatar')
    models.User.objects.create(username=name,avatar=avatar)
    return HttpResponse('ok') 
  return render(request,'upload.html')

总结

上面已经实现了将文件上传到数据库的功能,需要注意的有几点:
 1.所谓的上传到数据库,不是讲图片本身或者二进制码放在数据库,实际上也是将文件上传到服务器本地,数据库只是存了一个文件的路径,这样用户要调用文件的时候就可以通过路径去服务器指定的位置找了
 2.创建ORM的时候,avatar字段要有一个upload_to=''的属性,指定上传后的文件放在哪里
 3.往数据库添加的时候,文件字段属性赋值跟普通字段在形式上是一样的,如:models.User.objects.create(username=name,avatar=avatar)
 4.如果有两个用户上传的文件名重复,系统会自动将文件改名,效果如下:

Django项目实战之用户头像上传与访问的示例

附加

功能我们是实现了,看起来我们在调用文件的时候,只需要通过数据库文件路径已经保存的文件本身就可以访问图片,让它出现在网页上,其实并不是这样,

我们需要配置一些东西,django才可以找的到,不然的话就会过不了urls验证,而我们之所以可以直接访问static里的静态文件,是因为django已经帮我们配置好了。

Django项目实战之用户头像上传与访问的示例

配置步骤如下:

1、在站点的setting.py里配置

MEDIA_ROOT=os.path.join(BASE_DIR,"blog","media") #blog是项目名,media是约定成俗的文件夹名

MEDIA_URL="/media/"   # 跟STATIC_URL类似,指定用户可以通过这个路径找到文件

2、在urls.py里配置

from django.views.static import serve
from upload import settings        #upload是站点名
url(r'^media/(?P<path>.*)$', serve, {'document_root': settings.MEDIA_ROOT}),

配置完后,就可以通过http://127.0.0.1:8001/media/milk.png访问到图片了 

3 用AJAX提交文件

upload.html

<!DOCTYPE html>

<html lang="en">
<head>
  <meta charset="UTF-8">
</head>
<body>
<form>
  {% csrf_token %}
  <div>用户名:<input id="name-input" type="text"></div>

  <div>头像<input id="avatar-input" type="file"></div>
  <input id="submit-btn" type="button" value="提交">
</form>
<script src="/static/js/jquery-3.2.1.min.js"></script>
<script>
  $('#submit-btn').on('click',function () {
    formdata = new FormData();
    formdata.append('username',$('#name-input').val());
    formdata.append("avatar",$("#avatar")[0].files[0]);
    formdata.append("csrfmiddlewaretoken",$("[name='csrfmiddlewaretoken']").val()); 
 $.ajax({
 processData:false,contentType:false,url:'/upload', type:'post', data:formdata,success:function (arg)
 { 
if (arg.state == 1){ alert('成功!') }
else { alert('失败!') } } }) });
 </script>
 </body> 
</html>

views.py

from django.shortcuts import render,HttpResponse
from django.http import JsonResponse
from app01 import models
def upload(request):
  if request.method == 'POST':
    name = request.POST.get('username')
    avatar = request.FILES.get('avatar')
    try:
      models.User.objects.create(username=name,avatar=avatar)
      data = {'state':1}
    except:
      data = {'state':0}
    return JsonResponse(data)
  return render(request,'upload.html')

总结

1.Ajax上传的时候,按钮的tpye一定不要用submit
2.Ajax上传的时候data参数的值不再是一个普通‘字典'类型的值,而是一个FormData对像

  1. 创建对象formdata = new FormData(); 
  2. 往里面添加值formdata.append('username',$('#name-input').val());

3.Ajax在做post提交的时候要加上csrf验证

  1. formdata.append("csrfmiddlewaretoken",$("[name='csrfmiddlewaretoken']").val());

4.最后,Ajax上传文件的时候要有两个参数设置

  1. processData:false
  2. contentType:false

 4 上传图片文件的时候有预览功能

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
</head>
<body>
<form>
  <!----用一个label标签将上传文件输入框跟图片绑定一起,
     点击图片的时候就相当于点击了上传文件的按钮---->
  <label><img id="avatar-img" src="/static/img/default.png" width="80px" height="80px">
    <div>头像<input id="avatar-input" hidden type="file"></div>

  </label>
  <input id="submit-btn" type="button" value="提交">
</form>
<script src="/static/js/jquery-3.2.1.min.js"></script>
<script>
  // 上传文件按钮(label里的图片)点击事件
  $('#avatar-input').on('change',function () {
    // 获取用户最后一次选择的图片
    var choose_file=$(this)[0].files[0];
    // 创建一个新的FileReader对象,用来读取文件信息
    var reader=new FileReader();
    // 读取用户上传的图片的路径
    reader.readAsDataURL(choose_file);
    // 读取完毕之后,将图片的src属性修改成用户上传的图片的本地路径
    reader.onload=function () {
       $("#avatar-img").attr("src",reader.result)
    }
  });
</script>

5 大总结

对于文件上传,不管是直接form提交也好,Ajax提交也好,根本问题是要告诉浏览器你要上传的是一个文件而不是普通的字符串

而怎么样告诉浏览器呢,就是通过请求体重的ContentType参数,我们上传普通的字符串的时候不用指定,因为它有默认值,

而如果要传文件的话,就要另外指定了。总结以下几点
 1.form表单上传的话是通过 enctype="multipart/form-data" 来指定ContentType
 2.ajax上传的话是通过  processData:false 和 contentType:false来指定ContentType
 3.form上传的时候,文件数据是通过<input type="file">标签来‘'包裹‘'数据,
 4.ajax上传的时候,是通过一个 FormData 实例对象来添加数据,传递的时候传递这个对象就行了
 5.数据传递过去之后,是封装在request.FILES里,而不是request.POST里

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
使用Python程序抓取新浪在国内的所有IP的教程
May 04 Python
举例讲解如何在Python编程中进行迭代和遍历
Jan 19 Python
解决python2.7用pip安装包时出现错误的问题
Jan 23 Python
node.js获取参数的常用方法(总结)
May 29 Python
Python常见异常分类与处理方法
Jun 04 Python
python编写简易聊天室实现局域网内聊天功能
Jul 28 Python
Django 路由系统URLconf的使用
Oct 11 Python
python实现维吉尼亚加密法
Mar 20 Python
pyqt 实现QlineEdit 输入密码显示成圆点的方法
Jun 24 Python
python pytest进阶之xunit fixture详解
Jun 27 Python
python 制作网站筛选工具(附源码)
Jan 21 Python
Python机器学习三大件之一numpy
May 10 Python
基于Python 装饰器装饰类中的方法实例
Apr 21 #Python
使用python装饰器计算函数运行时间的实例
Apr 21 #Python
Python实现针对给定字符串寻找最长非重复子串的方法
Apr 21 #Python
Python 实现一行输入多个值的方法
Apr 21 #Python
Python实现接受任意个数参数的函数方法
Apr 21 #Python
深入分析python数据挖掘 Json结构分析
Apr 21 #Python
Python编程中NotImplementedError的使用方法
Apr 21 #Python
You might like
咖啡磨器 如何选购一台适合家用的意式磨豆机
2021/03/05 新手入门
简单的PHP缓存设计实现代码
2011/09/30 PHP
PHP mysql与mysqli事务使用说明 分享
2013/08/17 PHP
php实现12306火车票余票查询和价格查询(12306火车票查询)
2014/01/14 PHP
使用ob系列函数实现PHP网站页面静态化
2014/08/13 PHP
php图像处理类实例
2015/07/28 PHP
php操作redis数据库常见方法实例总结
2020/02/20 PHP
JavaScript null和undefined区别分析
2009/10/14 Javascript
js 父窗口控制子窗口的行为-打开,关闭,重定位,回复
2010/04/20 Javascript
JQuery slideshow的一个小问题(如何发现及解决过程)
2013/02/06 Javascript
简单易用的倒计时js代码
2014/08/04 Javascript
JS应用正则表达式转换大小写示例
2014/09/18 Javascript
JavaScript ES6中CLASS的使用详解
2016/11/22 Javascript
原生js实现无缝轮播图效果
2017/01/11 Javascript
使用store来优化React组件的方法
2017/10/23 Javascript
JavaScript代码实现txt文件的上传预览功能
2018/03/27 Javascript
使用Python的Scrapy框架编写web爬虫的简单示例
2015/04/17 Python
微信跳一跳python辅助软件思路及图像识别源码解析
2018/01/04 Python
Django中反向生成models.py的实例讲解
2018/05/30 Python
如何在Django中设置定时任务的方法示例
2019/01/18 Python
python实现数据分析与建模
2019/07/11 Python
对django后台admin下拉框进行过滤的实例
2019/07/26 Python
Python利用全连接神经网络求解MNIST问题详解
2020/01/14 Python
Python图像阈值化处理及算法比对实例解析
2020/06/19 Python
Python计算矩阵的和积的实例详解
2020/09/10 Python
KEETSA环保床垫:更好的睡眠,更好的生活!
2016/11/24 全球购物
MANGO官方网站:西班牙芒果服装品牌
2017/01/15 全球购物
苏格兰领先的多渠道鞋店:Begg Shoes
2019/10/22 全球购物
Jdbc数据访问技术面试题
2012/03/30 面试题
计算机应用专业学生的自我评价分享
2013/11/03 职场文书
国际商务专业职业生涯规划书范文
2014/01/17 职场文书
餐饮企业总经理岗位职责范文
2014/02/18 职场文书
初中班长竞选稿
2015/11/20 职场文书
2016秋季小学开学寄语
2015/12/03 职场文书
React Native项目框架搭建的一些心得体会
2021/05/28 Javascript
电脑关机速度很慢怎么办 提升电脑关机速度设置教程
2022/04/08 数码科技