校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁(yè) > 知識(shí)庫(kù) > django上傳文件的三種方式

django上傳文件的三種方式

熱門標(biāo)簽:悟空智電銷機(jī)器人6 外呼不封號(hào)系統(tǒng) 電信營(yíng)業(yè)廳400電話申請(qǐng) 幫人做地圖標(biāo)注收費(fèi)算詐騙嗎 蘇州電銷機(jī)器人十大排行榜 溫州旅游地圖標(biāo)注 江蘇房產(chǎn)電銷機(jī)器人廠家 荊州云電銷機(jī)器人供應(yīng)商 遼寧400電話辦理多少錢

Django文件上傳需要考慮的重要事項(xiàng)

文件或圖片一般通過(guò)表單進(jìn)行。用戶在前端點(diǎn)擊文件上傳,然后以POST方式將數(shù)據(jù)和文件提交到服務(wù)器。服務(wù)器在接收到POST請(qǐng)求后需要將其存儲(chǔ)在服務(wù)器上的某個(gè)地方。Django默認(rèn)的存儲(chǔ)地址是相對(duì)于根目錄的/media/文件夾,存儲(chǔ)的默認(rèn)文件名就是文件本來(lái)的名字。上傳的文件如果不大于2.5MB,會(huì)先存入服務(wù)器內(nèi)存中,然后再寫入磁盤。如果上傳的文件很大,Django會(huì)把文件先存入臨時(shí)文件,再寫入磁盤。

Django默認(rèn)處理方式會(huì)出現(xiàn)一個(gè)問(wèn)題,所有文件都存儲(chǔ)在一個(gè)文件夾里。不同用戶上傳的有相同名字的文件可能會(huì)相互覆蓋。另外用戶還可能上傳一些不安全的文件如js和exe文件,我們必需對(duì)允許上傳文件的類型進(jìn)行限制。因此我們?cè)诶肈jango處理文件上傳時(shí)必需考慮如下3個(gè)因素:

  • 設(shè)置存儲(chǔ)上傳文件的文件夾地址
  • 對(duì)上傳文件進(jìn)行重命名
  • 對(duì)可接受的文件類型進(jìn)行限制(表單驗(yàn)證)

注意:以上事項(xiàng)對(duì)于上傳圖片是同樣適用的。

Django文件上傳的3種常見(jiàn)方式

Django文件上傳一般有3種方式(如下所示)。我們會(huì)針對(duì)3種方式分別提供代碼示范。

  • 使用一般的自定義表單上傳,在視圖中手動(dòng)編寫代碼處理上傳的文件
  • 使用由模型創(chuàng)建的表單(ModelForm)上傳,使用form.save()方法自動(dòng)存儲(chǔ)
  • 使用Ajax實(shí)現(xiàn)文件異步上傳,上傳頁(yè)面無(wú)需刷新即可顯示新上傳的文件

Ajax文件上傳部分見(jiàn)Django與Ajax交互篇。

項(xiàng)目創(chuàng)建與設(shè)置

我們先使用django-admin startproject命令創(chuàng)建一個(gè)叫file_project的項(xiàng)目,然后cd進(jìn)入file_project, 使用python manage.py startapp創(chuàng)建一個(gè)叫file_upload的app。

我們首先需要將file_upload這個(gè)app加入到我們項(xiàng)目里,然后設(shè)置/media/和/STATIC_URL/文件夾。我們上傳的文件都會(huì)放在/media/文件夾里。我們還需要使用css和js這些靜態(tài)文件,所以需要設(shè)置STATIC_URL。

#file_project/settings.py

INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    'file_upload',# 新增
]

STATIC_URL = '/static/'
STATICFILES_DIRS = [os.path.join(BASE_DIR, "static"), ]

MEDIA_ROOT = os.path.join(BASE_DIR, 'media')
MEDIA_URL = '/media/'

#file_project/urls.py
from django.contrib import admin
from django.urls import path, include
from django.conf import settings
from django.conf.urls.static import static

urlpatterns = [
    path('admin/', admin.site.urls),
    path('file/', include("file_upload.urls")),
] + static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)

創(chuàng)建模型

使用Django上傳文件創(chuàng)建模型不是必需,然而如果我們需要對(duì)上傳文件進(jìn)行系統(tǒng)化管理,模型還是很重要的。我們的File模型包括file和upload_method兩個(gè)字段。我們通過(guò)upload_to選項(xiàng)指定了文件上傳后存儲(chǔ)的地址,并對(duì)上傳的文件名進(jìn)行了重命名。

#file_upload/models.py
from django.db import models
import os
import uuid

# Create your models here.
# Define user directory path
def user_directory_path(instance, filename):
    ext = filename.split('.')[-1]
    filename = '{}.{}'.format(uuid.uuid4().hex[:10], ext)
    return os.path.join("files", filename)

class File(models.Model):
    file = models.FileField(upload_to=user_directory_path, null=True)
    upload_method = models.CharField(max_length=20, verbose_name="Upload Method")

注意:如果你不使用ModelForm,你還需要手動(dòng)編寫代碼存儲(chǔ)上傳文件。

URLConf配置

本項(xiàng)目一共包括3個(gè)urls, 分別對(duì)應(yīng)普通表單上傳,ModelForm上傳和顯示文件清單。

#file_upload/urls.py
from django.urls import re_path, path
from . import views

# namespace
app_name = "file_upload"

urlpatterns = [
    # Upload File Without Using Model Form
    re_path(r'^upload1/$', views.file_upload, name='file_upload'),

    # Upload Files Using Model Form
    re_path(r'^upload2/$', views.model_form_upload, name='model_form_upload'),

    # View File List
    path('file/', views.file_list, name='file_list'),

]

使用一般表單上傳文件

我們先定義一個(gè)一般表單FileUploadForm,并通過(guò)clean方法對(duì)用戶上傳的文件進(jìn)行驗(yàn)證,如果上傳的文件名不以jpg, pdf或xlsx結(jié)尾,將顯示表單驗(yàn)證錯(cuò)誤信息。關(guān)于表單的自定義和驗(yàn)證更多內(nèi)容見(jiàn)Django基礎(chǔ): 表單forms的設(shè)計(jì)與使用。

#file_upload/forms.py

from django import forms
from .models import File

# Regular form
class FileUploadForm(forms.Form):
    file = forms.FileField(widget=forms.ClearableFileInput(attrs={'class': 'form-control'}))
    upload_method = forms.CharField(label="Upload Method", max_length=20,
                                   widget=forms.TextInput(attrs={'class': 'form-control'}))
    def clean_file(self):
        file = self.cleaned_data['file']
        ext = file.name.split('.')[-1].lower()
        if ext not in ["jpg", "pdf", "xlsx"]:
            raise forms.ValidationError("Only jpg, pdf and xlsx files are allowed.")
        # return cleaned data is very important.
        return file

注意: 使用clean方法對(duì)表單字段進(jìn)行驗(yàn)證時(shí),別忘了return驗(yàn)證過(guò)的數(shù)據(jù),即cleaned_data。只有返回了cleaned_data, 視圖中才可以使用form.cleaned_data.get(‘xxx')獲取驗(yàn)證過(guò)的數(shù)據(jù)。

對(duì)應(yīng)一般文件上傳的視圖file_upload方法如下所示。當(dāng)用戶的請(qǐng)求方法為POST時(shí),我們通過(guò)form.cleaned_data.get('file')獲取通過(guò)驗(yàn)證的文件,并調(diào)用自定義的handle_uploaded_file方法來(lái)對(duì)文件進(jìn)行重命名,寫入文件。如果用戶的請(qǐng)求方法不為POST,則渲染一個(gè)空的FileUploadForm在upload_form.html里。我們還定義了一個(gè)file_list方法來(lái)顯示文件清單。

#file_upload/views.py

from django.shortcuts import render, redirect
from .models import File
from .forms import FileUploadForm, FileUploadModelForm
import os
import uuid
from django.http import JsonResponse
from django.template.defaultfilters import filesizeformat

# Create your views here.


# Show file list
def file_list(request):
    files = File.objects.all().order_by("-id")
    return render(request, 'file_upload/file_list.html', {'files': files})

# Regular file upload without using ModelForm
def file_upload(request):
    if request.method == "POST":
        form = FileUploadForm(request.POST, request.FILES)
        if form.is_valid():
            # get cleaned data
            upload_method = form.cleaned_data.get("upload_method")
            raw_file = form.cleaned_data.get("file")
            new_file = File()
            new_file.file = handle_uploaded_file(raw_file)
            new_file.upload_method = upload_method
            new_file.save()
            return redirect("/file/")
    else:
        form = FileUploadForm()

    return render(request, 'file_upload/upload_form.html', 
                  {'form': form, 'heading': 'Upload files with Regular Form'}
                 )

def handle_uploaded_file(file):
    ext = file.name.split('.')[-1]
    file_name = '{}.{}'.format(uuid.uuid4().hex[:10], ext)

    # file path relative to 'media' folder
    file_path = os.path.join('files', file_name)
    absolute_file_path = os.path.join('media', 'files', file_name)

    directory = os.path.dirname(absolute_file_path)
    if not os.path.exists(directory):
        os.makedirs(directory)

    with open(absolute_file_path, 'wb+') as destination:
        for chunk in file.chunks():
            destination.write(chunk)

    return file_path

注意:

  • handle_uploaded_file方法里文件寫入地址必需是包含/media/的絕對(duì)路徑,如果/media/files/xxxx.jpg,而該方法返回的地址是相對(duì)于/media/文件夾的地址,如/files/xxx.jpg。存在數(shù)據(jù)中字段的是相對(duì)地址,而不是絕對(duì)地址。
  • 構(gòu)建文件寫入絕對(duì)路徑時(shí)請(qǐng)用os.path.join方法,因?yàn)椴煌到y(tǒng)文件夾分隔符不一樣。寫入文件前一個(gè)良好的習(xí)慣是使用os.path.exists檢查目標(biāo)文件夾是否存在,如果不存在先創(chuàng)建文件夾,再寫入。

上傳表單模板upload_form.html代碼如下:

#file_upload/templates/upload_form.html
{% extends "file_upload/base.html" %}
{% block content %}
{% if heading %}
h3>{{ heading }}/h3>
{% endif %}

form action="" method="post" enctype="multipart/form-data" >
  {% csrf_token %}
  {{ form.as_p }}
 button class="btn btn-info form-control " type="submit" value="submit">Upload/button>
/form>
{% endblock %} 

顯示文件清單模板file_list.html代碼如下所示:

# file_upload/templates/file_list.html
{% extends "file_upload/base.html" %}

{% block content %}
h3>File List/h3>
p> a href="/file/upload1/" rel="external nofollow" >RegularFormUpload/a> | a href="/file/upload2/" rel="external nofollow" >ModelFormUpload/a>
    | a href="/file/upload3/" rel="external nofollow" >AjaxUpload/a>/p>
{% if files %}
table class="table table-striped">
    tbody>
    tr>
        td>Filename  URL/td>
        td>Filesize/td>
        td>Upload Method/td>
    /tr>
    {% for file in files %}
    tr>
        td>a href="{{ file.file.url }}" rel="external nofollow" >{{ file.file.url }}/a>/td>
        td>{{ file.file.size | filesizeformat }}/td>
        td>{{ file.upload_method }}/td>
    /tr>
    {% endfor %}
    /tbody>
/table>

{% else %}

p>No files uploaded yet. Please click a href="{% url 'file_upload:file_upload' %}" rel="external nofollow" >here/a>
    to upload files./p>
{% endif %}
{% endblock %}

注意: 

  • 對(duì)于上傳的文件我們可以調(diào)用file.url, file.name和file.size來(lái)查看上傳文件的鏈接,地址和大小。
  • 上傳文件的大小默認(rèn)是以B顯示的,數(shù)字非常大。使用Django模板過(guò)濾器filesizeformat可以將文件大小顯示為人們可讀的方式,如MB,KB。

使用ModelForm上傳文件

使用ModelForm上傳是小編我推薦的上傳方式,前提是你已經(jīng)在模型中通過(guò)upload_to選項(xiàng)自定義了用戶上傳文件存儲(chǔ)地址,并對(duì)文件進(jìn)行了重命名。我們首先要自定義自己的FileUploadModelForm,由File模型重建的。代碼如下所示:

#file_upload/forms.py
from django import forms
from .models import File

# Model form
class FileUploadModelForm(forms.ModelForm):
    class Meta:
        model = File
        fields = ('file', 'upload_method',)
        widgets = {
            'upload_method': forms.TextInput(attrs={'class': 'form-control'}),
            'file': forms.ClearableFileInput(attrs={'class': 'form-control'}),
        }

    def clean_file(self):
        file = self.cleaned_data['file']
        ext = file.name.split('.')[-1].lower()
        if ext not in ["jpg", "pdf", "xlsx"]:
            raise forms.ValidationError("Only jpg, pdf and xlsx files are allowed.")
        # return cleaned data is very important.
        return file

使用ModelForm處理文件上傳的視圖model_form_upload方法非常簡(jiǎn)單,只需調(diào)用form.save()即可,無(wú)需再手動(dòng)編寫代碼寫入文件。

#file_upload/views.py

from django.shortcuts import render, redirect
from .models import File
from .forms import FileUploadForm, FileUploadModelForm
import os
import uuid
from django.http import JsonResponse
from django.template.defaultfilters import filesizeformat

# Create your views here.
# Upload File with ModelForm

def model_form_upload(request):
    if request.method == "POST":
        form = FileUploadModelForm(request.POST, request.FILES)
        if form.is_valid():
            form.save() # 一句話足以
            return redirect("/file/")
    else:
        form = FileUploadModelForm()

    return render(request, 'file_upload/upload_form.html', 
                  {'form': form,'heading': 'Upload files with ModelForm'}
                 )

模板跟前面一樣,這里就不展示了。

GitHub源碼地址

https://github.com/shiyunbo/django-file-upload-download

小結(jié)

本文提供并解讀了利用Django上傳文件的3種主要方式(一般表單上傳,ModelForm上傳和Ajax上傳)及示范代碼。

以上就是django上傳文件的三種方式的詳細(xì)內(nèi)容,更多關(guān)于django上傳文件的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

您可能感興趣的文章:
  • Django 如何實(shí)現(xiàn)文件上傳下載
  • Django和Ueditor自定義存儲(chǔ)上傳文件的文件名
  • 詳解Django自定義圖片和文件上傳路徑(upload_to)的2種方式
  • 基于django和dropzone.js實(shí)現(xiàn)上傳文件
  • python中Django文件上傳方法詳解
  • Django后端分離 使用element-ui文件上傳方式
  • Django Admin 上傳文件到七牛云的示例代碼
  • Django實(shí)現(xiàn)任意文件上傳(最簡(jiǎn)單的方法)
  • Django 解決上傳文件時(shí),request.FILES為空的問(wèn)題
  • Django中文件上傳和文件訪問(wèn)微項(xiàng)目的方法
  • django 文件上傳功能的相關(guān)實(shí)例代碼(簡(jiǎn)單易懂)

標(biāo)簽:臺(tái)灣 宿遷 欽州 景德鎮(zhèn) 濟(jì)南 喀什 三沙 黃山

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《django上傳文件的三種方式》,本文關(guān)鍵詞  django,上傳,文件,的,三種,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《django上傳文件的三種方式》相關(guān)的同類信息!
  • 本頁(yè)收集關(guān)于django上傳文件的三種方式的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃
    91精品国产综合久久久蜜臀图片| 欧美国产日韩一二三区| 亚洲精品国产品国语在线app| 日本亚洲最大的色成网站www| 色网综合在线观看| 亚洲伦理在线精品| 7777精品伊人久久久大香线蕉的 | 国产精品888| 欧美国产一区在线| 一本久道中文字幕精品亚洲嫩| 日韩伦理电影网| 欧美精品v国产精品v日韩精品| 日韩精品福利网| 久久久蜜桃精品| 成人黄色av网站在线| 日韩久久一区二区| 欧美精品第1页| 国产伦精品一区二区三区视频青涩 | 欧美大片日本大片免费观看| 成人激情开心网| 一区二区三区在线视频播放| 在线观看免费视频综合| 久久疯狂做爰流白浆xx| 亚洲精品一区二区三区99| jizzjizzjizz欧美| 亚洲午夜精品久久久久久久久| 欧美精品电影在线播放| 国产一区二区三区香蕉| 亚洲精品国产高清久久伦理二区| 欧美日韩在线观看一区二区 | 欧美老女人第四色| 国产伦理精品不卡| 亚洲免费av高清| 日韩欧美国产系列| 欧美性色综合网| 成人av第一页| 亚洲一区二区三区中文字幕| 国产亚洲视频系列| 欧美福利电影网| 色哟哟亚洲精品| 国产精品一区专区| 亚洲综合在线观看视频| 日韩精品一区二区三区三区免费 | 国产精品久久三| 7878成人国产在线观看| 91丨九色丨黑人外教| 麻豆高清免费国产一区| 亚洲成av人片www| √…a在线天堂一区| 久久久久久久久99精品| 欧美日韩另类一区| 成人h版在线观看| 国内一区二区在线| 国产一区二区三区电影在线观看| 亚洲电影中文字幕在线观看| 国产精品久久久久久久午夜片| 欧美大胆一级视频| 日韩一区二区三区三四区视频在线观看| 97久久人人超碰| 99国内精品久久| 91麻豆精品国产91| 91麻豆精品国产自产在线观看一区| 欧美日韩三级一区| 在线不卡一区二区| 777久久久精品| 欧美日韩国产一二三| 日本精品视频一区二区| 99国产精品久| 国产精品456露脸| 成人国产免费视频| 欧美视频精品在线观看| 在线成人小视频| 精品成a人在线观看| 国产精品久久久久久久久免费桃花 | 狠狠色狠狠色综合| 丰满少妇在线播放bd日韩电影| 粉嫩av亚洲一区二区图片| 色偷偷久久一区二区三区| 欧美日韩一区二区三区四区| 久久午夜老司机| 亚洲第一搞黄网站| 粉嫩绯色av一区二区在线观看| 欧美体内she精高潮| 欧美国产精品一区二区三区| 亚洲在线免费播放| 国产iv一区二区三区| 欧美色视频在线观看| 亚洲欧洲日韩女同| 老司机免费视频一区二区三区| www.亚洲国产| 久久综合国产精品| 美女免费视频一区二区| 在线观看视频一区二区欧美日韩| 2020国产精品自拍| 亚洲va欧美va人人爽午夜| 成人视屏免费看| 精品国产免费人成在线观看| 天天综合天天综合色| 波多野结衣中文字幕一区二区三区| 欧美乱妇15p| 一区二区三区欧美激情| 99久久国产免费看| 一区在线观看免费| 成人伦理片在线| 日韩欧美123| 国产成人精品网址| 欧美国产1区2区| 不卡一二三区首页| 亚洲视频一二三| 91麻豆.com| 亚洲电影激情视频网站| 欧美日韩在线一区二区| 欧美aa在线视频| 国产精品区一区二区三区| 精品在线免费观看| 久久奇米777| 成人涩涩免费视频| 亚洲免费在线看| 91精品视频网| 国产露脸91国语对白| 亚洲国产精品成人综合色在线婷婷| 丁香桃色午夜亚洲一区二区三区| 国产欧美久久久精品影院| 成人av综合一区| 亚洲一本大道在线| 欧美va亚洲va在线观看蝴蝶网| 国产精品1区2区| 国产免费成人在线视频| 欧美高清视频不卡网| 国产高清精品久久久久| 亚洲综合在线免费观看| 欧美体内she精高潮| 韩国一区二区三区| 亚洲一区二区三区四区在线观看 | 欧美日韩精品综合在线| 黄页视频在线91| 日韩国产欧美三级| 亚洲美女在线国产| 精品国产乱码久久久久久浪潮| 99精品久久只有精品| 精品一区二区三区在线观看国产| 1区2区3区欧美| 国产欧美日韩另类一区| 久久久www成人免费毛片麻豆| 欧美老肥妇做.爰bbww| 91老师国产黑色丝袜在线| 国产99久久久国产精品潘金| 久久精品国产一区二区| 日韩高清一级片| 亚洲欧美日韩一区二区| 国产精品剧情在线亚洲| 久久久三级国产网站| 国产亚洲婷婷免费| 久久久精品国产99久久精品芒果| 精品国产91亚洲一区二区三区婷婷 | 国产成人综合亚洲网站| 黑人精品欧美一区二区蜜桃| 七七婷婷婷婷精品国产| 九色|91porny| 99麻豆久久久国产精品免费优播| caoporn国产一区二区| 色综合久久综合网97色综合| 日本高清不卡视频| 欧美一区二区三区视频在线 | 欧美日韩精品专区| 欧美一区二区女人| 久久久亚洲精华液精华液精华液 | 国产色综合久久| 亚洲国产精品欧美一二99| 秋霞午夜av一区二区三区| 国产美女一区二区三区| 一本色道**综合亚洲精品蜜桃冫| 一本一道久久a久久精品综合蜜臀| 色拍拍在线精品视频8848| 在线综合视频播放| 欧美国产精品一区二区| 日韩激情av在线| 成人动漫一区二区三区| 日韩一区国产二区欧美三区| 日韩一区在线看| 极品少妇一区二区三区精品视频 | 国产综合色视频| 欧美剧在线免费观看网站| 国产日韩av一区| 免费看日韩精品| 欧美日韩成人综合| 亚洲在线成人精品| 国产999精品久久久久久| 欧美日韩免费观看一区三区| 中文字幕一区二区三区在线不卡| 精品一区二区在线观看| 欧美日韩黄色影视| 国产精品婷婷午夜在线观看| 久久99这里只有精品| 3d动漫精品啪啪1区2区免费| 一区二区三区精品| 91在线观看成人| 亚洲欧美一区二区在线观看| 99久久99精品久久久久久 | 99在线精品观看|