首页前端开发VUEvue与django(drf)实现文件上传下载功能全过程

vue与django(drf)实现文件上传下载功能全过程

时间2024-02-11 05:04:03发布访客分类VUE浏览948
导读:收集整理的这篇文章主要介绍了vue与django(drf 实现文件上传下载功能全过程,觉得挺不错的,现在分享给大家,也给大家做个参考。 目录文件上传功能上传后端部分上传前端部分(vue添...
收集整理的这篇文章主要介绍了vue与django(drf)实现文件上传下载功能全过程,觉得挺不错的,现在分享给大家,也给大家做个参考。
目录
  • 文件上传功能
    • 上传后端部分
    • 上传前端部分(vue添加vue.js和node.js,设置eslint)
  • 文件下载功能
    • 下载后端部分
    • 下载前端部分
  • 总结

    文件上传功能

    上传后端部分

    (一)Models.py

    From django.db import modelsclass FilESModel(models.Model):                         //模型名(默认表名)	name = models.CharField(max_length=20, default='')      //name,fle是字段名(file为上传的文件)    file = models.FileField(upload_to='uploads/')         //upload上传功能实现,to上传后保存的路径    class Meta:        db_table = 'files_storage'             //自定义的表名        ordering = ['-id']                     //按顺序排列

    (二)Serializer.py

    使用 Django REST framework 实现后端 REST API,需创建序列化器 serializers.py,内容如下:

    from rest_framework import serializersfrom files import models                           //files 是 app 的名字class FilesSerializer(serializers.ModelSerializer):    class Meta:        model = models.FilesModel                      //指定模型        fields = '__all__'                             //指定序列化的字段名

    (三)views.py

    from rest_framework.viewsets import ModelViewSetfrom files import models, serializersclass FileViewSet(ModelViewSet):    queryset = models.FilesModel.objects.all()             //返回全部字段    serializer_class = serializers.FilesSerializer         //指定序列化器类型

    (四)urls

    1.项目总配置路径下(settings.py 所在的路径)编辑根路由配置文件 urls.py

    from django.contrib import adminfrom django.urls import path, includeurlpatterns = [    path('admin/', admin.sITe.urls),    path('storage/', include('files.urls'))         //网址前缀及指定某子app的url]

    2.app为files 的路径下新建 urls.py 文件,填写路由配置:

    from django.urls import include, pathfrom rest_framework import routersfrom files import viewsrouter = routers.DefaultRouter()router.register(r'files', views.FileViewSet)                   //注册路径urlpatterns = [    path('', include(router.urls))]

    (五)测试后端api

    运行后台服务 python manage.py runserver 0.0.0.0:8000,访问 http://xx.xx.xx.xx:8000/storage/files/,界面如下:

    上传前端部分(vue添加vue.js和node.js,设置eslint)

    template>
          div>
            el-label>
        名称/el-label>
            el-input v-model="fileData.name" style="width: 20%" />
            el-upload      ref="upload"      Drag      class="upload-demo"      action="http://xx.xx.xx.xx:8000/storage/files/"      :data="fileData"      :auto-upload="false"      :on-success="onSuccess"      style="padding: 30px"    >
              i class="el-icon-upload" />
              div class="el-upload__text">
        将文件拖到此处,或em>
        点击上传/em>
        /div>
            /el-upload>
            el-button style="margin-left: 10px;
        " size="small" tyPE="success" @click="submitUpload">
        上传到服务器/el-button>
          /div>
        /template>
        script>
    export default {
      name: 'UploadDemo',  data() {
        return {
          fileData: {
            name: ''      }
        }
      }
    ,  methods: {
        submitUpload() {
          this.$refs.upload.submit()    }
    ,    onSuccess() {
          this.$message.success('上传成功')    }
      }
    }
        /script>
        

    其中 el-upload 组件的 action 属性用于指定后台 API 的 URI;

    :auto-upload 属性用于设置是否自动上传(这里设置为 false,手动触发上传动作);

    :on-success 属性用于指定上传成功后触发的方法。

    submitUpload() 中的 this.$refs.upload.submit() 方法触发文件上传动作。

    其中 el-upload 组件的 :data 属性用于指定文件上传时附加的数据(类型为 JavaScript 对象)。

    注意:

    1.env.development文件里改成BASE_API = ‘/api’

    2.除增删改查外,上传的接口在index.vue文件里写了,不用额外在api文件夹里加接口

    3.route里的函数调用后端接口

    4.前端一个页面可对应后端多个接口

    上传完成,后台数据如下:

    [    {
            "file": "http://172.20.23.34:8000/storage/files/uploads/AnyDesk.exe",        "id": 19,        "name": "测试文件"    }
    ,    {
            "file": "http://172.20.23.34:8000/storage/files/uploads/template.htML",        "id": 18,        "name": ""    }
    ,    {
            "file": "http://172.20.23.34:8000/storage/files/uploads/20171215091830_55126_hSnPtZR.png",        "id": 17,        "name": ""    }
        ]

    文件下载功能

    下载后端部分

    views.py

        @action(methods=['get', 'post'], detail=True)    def download(self, request, pk=None, *args, **kwargs):        file_obj = self.get_object()        response = FileResponse(open(file_obj.file.path, 'rb'))        return response

    下载前端部分

    download.vue

    template>
          el-table    :data="filelist.filter(data =>
         !seArch || data.name.toLowerCase().includes(search.toLowerCase()))"    style="width: 100%"  >
            el-table-column      label="序号"      PRop="id"    />
            el-table-column      label="上传时间"      prop="date"    />
            el-table-column      label="上传用户"      prop="auther"    />
            el-table-column      label="文件名"      prop="filename"    />
            el-table-column      align="right"    >
              template slot="header">
                el-input          v-model="search"          size="mini"          placeholder="输入关键字搜索"        />
              /template>
              template slot-scope="scope">
                el-button          size="mini"          @click="handleDown(scope.$index, scope.row)"        >
        Down/el-button>
                el-button          size="mini"          type="danger"          @click="handleDelete(scope.$index, scope.row)"        >
        Delete/el-button>
              /template>
            /el-table-column>
          /el-table>
        /template>
        script>
    import {
     getPkgsList }
     from '@/api/user'import {
     getToken }
     from '@/utils/auth'import {
     delpkg }
     from '@/api/user'import axios from 'axios'export default {
      data() {
        return {
          headers: {
            Authorization: 'Bearer ' + getToken()      }
    ,      // adDForm: {
          //   // 文件的数组      //   pics: []      // }
    ,      filelist: [],      search: '',      listLoading: true    }
      }
    ,  created() {
        this.fetchPkgsList()  }
    ,  methods: {
        handleChange(file, filelist) {
          this.filelist = filelist.slice(-3)    }
    ,    onSuccess() {
          this.$message.success('上传成功')    }
    ,    fetchPkgsList() {
              this.listLoading = true      getPkgsList().then(response =>
     {
                console.LOG('getPkgsList ========>
     ', response)        this.filelist = response.data.results        this.listLoading = false      }
    )    }
    ,    downloadFile(url, options = {
    }
    ) {
              return new Promise((resolve, reject) =>
     {
            // console.log(`${
    url}
         请求数据,参数=>
    `, JSON.stringify(options))        axios.defaults.headers['Authorization'] = 'Bearer ' + getToken()        axios({
              method: 'post',          url: url, // 请求地址          data: options, // 参数          responseType: 'blob' // 表明返回服务器返回的数据类型        }
        ).then(          response =>
     {
                // console.log("下载响应",response)            resolve(response.data.result)            debugger            const blob = new Blob([response.data])            // console.log(blob)            // let fileName = Date.parse(new Date()) + '.xlsx'            // 切割出文件名            const fileNameEncode = options.filename            // 解码            const fileName = decodeURIcomponent(fileNameEncode)            // console.log("fileName",fileName)            if (window.navigator.msSaveOrOpenBlob) {
                  // console.log(2)              navigator.msSaveBlob(blob, fileName)            }
     else {
                  // console.log(3)              VAR link = document.createElement('a')              link.href = window.URL.createObjectURL(blob)              link.download = fileName              link.click()              // 释放内存              window.URL.revokeObjectURL(link.href)            }
              }
        ,          err =>
     {
                reject(err)          }
            )      }
    )    }
    ,    handleDown(index, row) {
          const postUrl = '/api/files/' + (row.id) + '/download/'      const params = {
     filename: row.filename }
          this.downloadFile(postUrl, params)    }
    ,    handleDelete(index, row) {
              this.listLoading = true      // const i = this.addForm.pics.findIndex((x) =>
     x.pic === row.file)      // this.addForm.pics.splice(i, 1)      this.$confirm(`确定移除 ${
    row.filename}
        ?`)      delpkg(row.id).then(response =>
     {
                // console.log('getPkgsList ========>
     ', response)        // console.log(index)        // console.log(this.filelist.length)        this.filelist = this.filelist.slice(index, 1)        this.fetchPkgsList()        // console.log(this.filelist)        this.listLoading = false      }
    )    }
      }
    }
        /script>
        

    总结

    到此这篇关于vue与django(drf)实现文件上传下载功能的文章就介绍到这了,更多相关vue与django文件上传下载内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!

    您可能感兴趣的文章:
    • Django实现文件上传下载功能
    • Django实现文件上传和下载功能
    • Django实现文件上传下载
    • Django文件上传与下载(FileFlid)
    • Django 如何实现文件上传下载
    • Django学习之文件上传与下载

    声明:本文内容由网友自发贡献,本站不承担相应法律责任。对本内容有异议或投诉,请联系2913721942#qq.com核实处理,我们将尽快回复您,谢谢合作!


    若转载请注明出处: vue与django(drf)实现文件上传下载功能全过程
    本文地址: https://pptw.com/jishu/609355.html
    vant-list组件触发多次onload事件导致数据乱序的解决方案 Vue3从0搭建Vite打包组件库使用详解

    游客 回复需填写必要信息