Django 文件上传(十二)

当 Django 处理文件上传时,文件数据最终会被放置在 request.FILES

查看文档:文件上传 | Django 文档 | Django

Django工程如下:

创建本地存储目录

在static/应用目录下创建uploads目录用于存储接收上传的文件

在settings.py 配置静态目录和上传目录

python 复制代码
#指定静态文件路径
STATICFILES_DIRS=[
    BASE_DIR / 'static'
]


#媒体文件本地存放路径 (如果是模块,则必须指定哦!!)
MEDIA_ROOT= BASE_DIR / 'App/static/uploads'

一、单个文件上传

1、models.py模型数据

python 复制代码
from django.db import models

# Create your models here.
#存放用户图片数据表
class UserModel(models.Model):
    #用户名
    name=models.CharField(max_length=100,unique=True)
    #图片
    icon=models.CharField(max_length=300)

    class Meta:
        db_table='user'
        verbose_name='用户表'
        verbose_name_plural=verbose_name

注意:迁移文件

2、upload.html页面

python 复制代码
<body>
     <h2>单个文件上传</h2>
     <hr/>
     <form action="" method="post" enctype="multipart/form-data">
         {% csrf_token %}
         <p>用户名:<input type="text" name="uname"/></p>
         <p>头像:<input type="file" name="icon"/></p>
         <p><button>上传图片</button></p>
     </form>
</body>

注意:from表单上传文件需要加 enctype="multipare/form-data" 必须是post请求。

3、views.py视图函数

python 复制代码
#单个文件上传
def upload_1(request):
    if request.method=='GET':
        return  render(request,'upload.html')
    elif request.method=='POST':
        #获取文件上传数据
        uname=request.POST.get('uname')
        #获取文件
        icon=request.FILES.get('icon')
        #上传到本地
        #调用方法-
        #337791d7-8ae5-4d97-b0d8-17dcc71fd94f.jpg
        filename=getUuidName()+icon.name[icon.name.rfind('.'):]

        #1.分段保存上传的路径
        file_path=os.path.join(settings.MEDIA_ROOT,filename)
        print(filename,file_path)
        #分段存入
        with open(file_path,'ab') as f:
            #循环
            for part in icon.chunks():
                f.write(part) #写入
                f.flush()  #清空缓存

        #2.将上传文件保存到数据表中
        user=UserModel()
        user.name=uname
        user.icon='uploads/'+filename
        user.save()

        return render(request, 'upload.html')


#通过uuid获取唯一的图片名
def getUuidName():
    return str(uuid.uuid4())  #随机生成名字

#显示图片
def showImg(request,id):
    user= UserModel.objects.get(pk=id)
    return render(request, 'show.html',{'user':user})

4、urls.py路由

python 复制代码
from django.contrib import admin
from django.urls import path
from App.views import *

urlpatterns = [
    path('index/', index),  # 静态文件
    path('up1/', upload_1),  # 文件上传
    path('show/<int:id>', showImg, name='show'),  # 显示图片
    path('up2/', upload_2),  # 文件上传more
    path('show2/<int:id>', showImg2, name='show2'),  # 显示图片more

    path('admin/', admin.site.urls),
]

5、show.html 显示上传的图片

python 复制代码
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    {%  load static %}
    <link rel="stylesheet" href="{% static 'css/show.css' %}">
</head>
<body>
    <h2>显示图片</h2>
    <hr/>
    {%  load static %}
    <p id="p">{{ user.name }}</p>
    <p><img src="{% static user.icon %}"> </p>
</body>
</html>

6、运行

二、多个文件上传

1、uploadmore.html页面

python 复制代码
<body>
     <h2>多个文件上传</h2>
     <hr/>
     <form action="" method="post" enctype="multipart/form-data">
         {% csrf_token %}
         <p>用户名:<input type="text" name="uname"/></p>
{#         <p>头像:<input type="file" name="icon"/></p>#}
         {{ form }}
         <p><button>上传图片</button></p>
     </form>
</body>

2、views.py视图函数

python 复制代码
#########################################################
#表单类
class FileUploadForm(forms.Form):
    files=forms.FileField(widget=forms.ClearableFileInput(attrs={'multiple':True}))


#多个文件上传
def upload_2(request):
    if request.method == 'POST':
        #获取表单类
        form=FileUploadForm(request.POST,request.FILES)
        #判断
        if form.is_valid():
            # 获取文件上传数据
            uname = request.POST.get('uname')
            # 获取文件
            icons = request.FILES.getlist('files')
            #保存多个文件名
            ll_names=[]
            #循环图片
            for file in icons:
                #处理上传文件
                handle_uploaded_file(file)
                #保存文件名
                ll_names.append('uploads/' + file.name)

            #列表转换成字符串
            strs = ','.join(ll_names)

            # 2.将上传文件保存到数据表中
            user = UserModel()
            user.name = uname
            user.icon =strs
            user.save()

            # print(ll_names, strs)

            return render(request, 'showmore.html')
    else:
        form = FileUploadForm()
    return render(request, 'uploadmore.html', {'form': form})



def handle_uploaded_file(file):
    """ 文件保存处理 """
    filePath = os.path.join(settings.MEDIA_ROOT, file.name)
    # 保存文件
    with open(filePath, 'wb+') as fp:
        for info in file.chunks():
            fp.write(info)
            fp.flush()

#显示图片
def showImg2(request,id):
    user= UserModel.objects.get(pk=id)
    return render(request, 'showmore.html',{'user':user})

3、自定义模板标签和过滤器

文档:https://docs.djangoproject.com/zh-hans/4.0/howto/custom-template-tags/

python 复制代码
from django.template import Library
register = Library()


@register.filter(name="split")
def split(value, key):
    """
        Returns the value turned into a list.
    """
    return value.split(key)

注意:必须重启服务器,自定义模板才生效!!

4、showmore.html 显示上传的图片

python 复制代码
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    {%  load static %}
    <link rel="stylesheet" href="{% static 'css/show.css' %}">
</head>
<body>
    <h2>显示更多图片</h2>
    <hr/>
    {%  load static %}
    <p id="p">{{ user.name }}</p>

{# 开发服务器并不会自动重启 添加 templatetags 模块后,你需要重启服务器,这样才能在模板中使用 tags 和 filters。#}
{#    加载自定义模板#}
{% load myfilter %}
    {% with user.icon|split:"," as details %}
         {% for im in details %}
            <p><img src="{% static im %}"> </p>
         {% endfor %}
  {% endwith %}

</body>
</html>

5、运行

单个或多个文件上传,查看文档还是容易实现!!!

相关推荐
coberup3 分钟前
django Forbidden (403)错误解决方法
python·django·403错误
龙哥说跨境35 分钟前
如何利用指纹浏览器爬虫绕过Cloudflare的防护?
服务器·网络·python·网络爬虫
monkey_meng35 分钟前
【Rust中的迭代器】
开发语言·后端·rust
余衫马38 分钟前
Rust-Trait 特征编程
开发语言·后端·rust
monkey_meng42 分钟前
【Rust中多线程同步机制】
开发语言·redis·后端·rust
小白学大数据1 小时前
正则表达式在Kotlin中的应用:提取图片链接
开发语言·python·selenium·正则表达式·kotlin
flashman9111 小时前
python在word中插入图片
python·microsoft·自动化·word
菜鸟的人工智能之路1 小时前
桑基图在医学数据分析中的更复杂应用示例
python·数据分析·健康医疗
懒大王爱吃狼2 小时前
Python教程:python枚举类定义和使用
开发语言·前端·javascript·python·python基础·python编程·python书籍
秃头佛爷3 小时前
Python学习大纲总结及注意事项
开发语言·python·学习