Django 如何實現文件上傳下載

1. 前言

大傢好,我是安果!

文件上傳、下載作為基礎功能,在 Web 項目中非常普遍,Django 項目如何實現文件上傳下載?

本篇文章將帶大傢 5 分鐘快速實現文件上傳下載功能

2. 實戰一下

詳細實現步驟如下( 9 步)

2-1  進入虛擬環境,創建一個項目及 App

workon django3

# 創建項目
django-admin startproject file_up_and_down_demo

# 進入項目根目錄
cd file_up_and_down_demo/

# 創建一個App
django-admin startapp index

2-2  創建模板目錄並配置 settings.py

在 index App 下創建一個 templates 文件夾,然後在項目配置文件 settings.py 中配置 App 及模板目錄

# settings.py

# 配置App
INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    'index',
]

TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates',
        'DIRS': [
            # 配置模板目錄
            os.path.join(BASE_DIR, 'index/templates')
        ],
        'APP_DIRS': True,
        'OPTIONS': {
            'context_processors': [
                'django.template.context_processors.debug',
                'django.template.context_processors.request',
                'django.contrib.auth.context_processors.auth',
                'django.contrib.messages.context_processors.messages',
            ],
        },
    },
]

2-3  創建文件模型,並映射到數據庫

以默認的 sqlite 為例,在 index App 下的 models.py 中自定義一個代表文件的模型

該模型包含 3 個字段:

  • 文件名稱
  • 文件保存路徑
  • 上傳時間
# index App models.py

from django.db import models
from django.utils import timezone


# 文件模型
class FileModel(models.Model):
    # 文件名稱
    name = models.CharField(max_length=50)

    # 文件保存路徑
    path = models.CharField(max_length=100)

    # 上傳時間
    upload_time = models.DateTimeField(default=timezone.now)

然後,在項目根目錄下執行下面 2 條命令,將模型結構映射到數據庫中

# 數據庫映射
Python3 manage.py makemigrations

python3 manage.py migrate

2-4  自定義表單控件

在 index App 下創建一個表單文件 forms.py

在內部自定義一個表單類,繼承於 forms.Form

# index App forms.py

from django import forms

class FileForm(forms.Form):
    file = forms.FileField(
        # 支持多文件上傳
        widget=forms.ClearableFileInput(attrs={'multiple': True}),
        label='請選擇文件',
    )

2-5  添加上傳、下載路由 URL

為上傳、下載功能添加路由 URL

# 項目urls.py
from django.contrib import admin
from django.urls import path, include

urlpatterns = [
    path('admin/', admin.site.urls),
    path('', include('index.urls'))
]

# index App urls.py
from django.urls import path

from .views import *

urlpatterns = [
    # 上傳
    path('', index_view, name='index'),

    # 下載
    path('download/<id>', download_view, name='download')
]

2-6  編寫模板文件

在 index App 的模板文件夾創建一個簡單的模板文件 upload.html

其中

  • form 代表視圖函數傳過來的表單實體對象
  • form.as_p 代表以字段格式渲染所有的表單元素
# index App upload.html
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>主頁-上傳文件</title>
</head>
<body>

<form method="post" enctype="multipart/form-data">
    {% csrf_token %}
    {{ form.as_p }}
    <input type="submit" value="確定上傳">
</form>

</body>
</html>

2-7  上傳視圖函數

在 index App 下的 views.py 中編寫上傳功能的視圖函數

需要註意的是,我們需要提前在項目根目錄創建一個 upload 文件夾,用於存放上傳的文件

# index App views.py

def index_view(request):
    """
    上傳文件
    :param request:
    :return:
    """
    if request.method == 'POST':
        form = FileForm(request.POST, request.FILES)
        if form.is_valid():
            # 選擇的文件
            files = request.FILES.getlist('file')

            # 遍歷寫入到數據庫中
            for file in files:
                # 寫入到數據庫中
                file_model = FileModel(name=file.name, path=os.path.join('./upload', file.name))
                file_model.save()

                # 寫入到服務器本地
                destination = open(os.path.join("./upload", file.name), 'wb+')
                for chunk in file.chunks():
                    destination.write(chunk)
                destination.close()

            # 提示上傳成功
            return HttpResponse('上傳成功!')
    else:
        form = FileForm()
        return render(request, 'upload.html', locals())

2-8  下載視圖函數

接著,編寫下載功能的視圖函數

# index App views.py

def download_view(request, id):
    """
    下載文件
    :param request:
    :param id:文件id
    :return:
    """
    file_result = FileModel.objects.filter(id=id)

    # 如果文件存在,就下載文件
    if file_result:

        file = list(file_result)[0]

        # 文件名稱及路徑
        name = file.name
        path = file.path

        # 讀取文件
        file = open(path, 'rb')
        response = FileResponse(file)

        # 使用urlquote對文件名稱進行編碼
        response['Content-Disposition'] = 'attachment;filename="%s"' % urlquote(name)

        return response
    else:
        return HttpResponse('文件不存在!')

2-9  運行並測試

運行項目,訪問下面的地址,並上傳一個文件

使用 Pycharm 打開 sqlite 數據庫,發現成功插入一條文件記錄,並且文件也上傳到 upload 文件夾下

接著訪問下面的地址實現文件下載功能「 其中,file_id 代表文件的 id 值 」

http://127.0.0.1:8000/download/file_id

3. 最後

文章通過一個簡單的例子實現瞭文件的上傳、下載功能,並同步文件記錄到數據庫

實際項目中,一般還包含文件列表、文件刪除等功能,這些功能隻需要結合數據庫來增刪查改即可實

代碼地址:https://github.com/xingag/python_web

以上就是Django 如何實現文件上傳下載的詳細內容,更多關於Django實現文件上傳下載的資料請關註WalkonNet其它相關文章!

推薦閱讀:

    None Found