nodejs实现(含前端代码)单文件的上传、删除、下载

文件的上传、删除、下载基本是每个项目都有的功能,免不了和后端小伙伴的对接。本文从前端的角度,实现简易的demo,帮助梳理这些功能前后端的交互流程。

1、页面功能描述

页面效果
  • 上传:只能单个文件上传,上传成功后,刷新文件列表。一个文件可以重复上传多次,但是内存中只存一份(以文件名唯一来判断,不考虑其它复杂情况,可以考虑存文件hash);
  • 删除:先删除在数据库中文件的上传记录,若该文件记录已经已经为0,那么从内存中删除该文件;
  • 下载:下载指定文件名的文件(get请求,利用a标签download属性);

2、前端部分

前端使用vuefile-saver,请提前npm install

2.1 前端整体代码(可能缺失部分css样式)

<template>
  <div class="container">
    <span class="loading" v-show="isLoading"></span>
    <p>
      <input type="file" id="file" @change="changeFile">
      <button type="button" @click="uploadFile">上传文件</button>
    </p>
    <table>
      <thead>
        <tr>
          <th>序号</th>
          <th>上传时间</th>
          <th>文件名</th>
          <th>文件大小</th>
          <th>操作</th>
        </tr>
      </thead>
      <tbody>
        <template v-if="fileList.length">
          <tr v-for="(item, index) in fileList">
            <td>{{index + 1}}</td>
            <td>{{item.dateTime}}</td>
            <td>{{item.fileName}}</td>
            <td>{{item.size}}</td>
            <td>
              <button type="button" @click="downloadFile(item.fileName)">下载</button>
              <button type="button" @click="deleteFile(item)" style="background-color: #e52050; margin-left: 10px;">删除</button>
            </td>
          </tr>
        </template>
        <template v-else>
          <tr>
            <td colspan="5" class="tc">无数据</td>
          </tr>
        </template>
      </tbody>
    </table>
  </div>
</template>

<script>
// 通过blob下都会先获取文件,然后生成文件,再下载 没有直接下载方式友好(直接下载会马上出来浏览器下载进度)。
// 在相同来源内使用URL只会使用a[download]
import {saveAs} from 'file-saver';
export default {
  name: 'FileList',
  data() {
    return {
      isLoading: true,
      fileName: '',
      fileList: []
    }
  },
  methods: {
    getFileList() {
      this.isLoading = true;
      // 设置了es的刷新间隔为1s 所以要用定时器。(后面看后端处理方式)
      setTimeout(() => {
        fetch('/api/fileList', {
          method: 'get'
        }).then(res => res.json())
          .then(data => {
            this.fileList = data.status === 200 ? data.data : [];
            this.isLoading = false;
          })
          .catch(error => {
            this.isLoading = false;
            console.error('Error:', error);
          });
      }, 1000);
    },
    deleteFile(item) {
      fetch(`/api/file?fileName=${item.fileName}&id=${item.id}`, {
        method: 'delete'
      }).then(res => res.json())
        .then(data => {
          data.status === 200 && this.getFileList();
        })
        .catch(error => console.error('Error:', error));
    },
    uploadFile() {
      if (!this.fileName) {
        window.alert('请选择需要上传的文件');
        return 0;
      }

      let formData = new FormData();
      let fileField = document.getElementById('file');
      formData.append('fileName', this.fileName);
      formData.append('file', fileField.files[0]);
      fetch('/api/file', {
        method: 'post',
        body: formData
      }).then(res => res.json())
        .then(data => {
          if (data.status === 200) {
            fileField.value = '';
            this.fileName = '';
            this.getFileList();
          }
        })
        .catch(error => console.error('Error:', error));
    },
    // 这种方式是把文件流读取到浏览器内存中
    downloadFileByFileSaver(url, method) {
      let fileName = url.split('fileName=')[1];
      fetch(url, {
        method
      }).then(res => res.blob())
        .then(data => {
          let blob = new Blob([data], {
            type: 'application/octet-stream'
          });
          saveAs(blob, fileName);
        })
        .catch(error => console.error('Error:', error));
    },
    downloadFileByIframe(url) {
      let iframe = document.createElement('iframe');
      iframe.style.display = 'none';
      iframe.src = url;
      document.body.appendChild(iframe);
      setTimeout(() => document.body.removeChild(iframe), 100);
    },
    downloadFileByAtagClick(url, fileName = '') {
      let a = document.createElement('a');
      a.style.display = 'none';
      a.href = url;
      document.body.appendChild(a);
      fileName && a.setAttribute('download', fileName);
      a.click();
      document.body.removeChild(a);

      // 方法二: saveAs(url);
    },
    // 这种方式是把文件流读取到浏览器内存中
    downloadFileByBlob(url, method = 'get') {
      // 使用这种方式多了创建请求的时间
      let fileName = url.split('fileName=')[1];
      fetch(url, {
        method
      }).then(res => {
        console.log(res.headers.get('content-length'));
        return res.blob();
      }).then(data => {
          let blob = new Blob([data], {
            type: 'application/octet-stream'
          });
          // url表示指定的 File 对象或 Blob 对象。
          let url = URL.createObjectURL(blob);
          this.downloadFileByAtagClick(url, fileName);
          // 释放URL对象
          URL.revokeObjectURL(url);
        })
        .catch(error => console.error('Error:', error));
    },
    downloadFile(fileName) {
      // this.downloadFileByIframe(`/api/file?fileName=${fileName}`);
      this.downloadFileByAtagClick(`/api/file?fileName=${fileName}`);
      // this.downloadFileByBlob(`/api/file?fileName=test3.zip`, 'get');
      // this.downloadFileByFileSaver(`/api/file?fileName=${fileName}`, 'get');
    },
    changeFile(e) {
      this.fileName = e.target.files.length ? e.target.files[0].name : '';
    }
  },
  mounted() {
    this.getFileList();
  }
}
</script>

<style scoped>
.loading {
  position: fixed;
  left: 50%;
  top: 50%;
  width: 150px;
  height: 100px;
  background: url("../assets/loading.gif");
  background-size: cover;
  transform: translateX(-50%) translateY(-50%);
}
th {
  padding: 10px;
  background-color: #60bb63;
  color: white;
  font-size: 15px;
}
td {
  padding: 6px 10px;
}
</style>

2.2 前端文件下载的几种方式

前端文件下载主要有2种方式:

  • 利用浏览器提供的get请求,来直接下载文件,前端常见处理方式为:a [href][download]iframe [src]
    选这种方式要看后端下载的实现方式,是否能让前端直接下载。这种方式比好的是下载是同步的,点击下载按钮后,能直接看到浏览器的反应。
  • 利用blob,获取后端文件流,再生成URL,利用a [href][download]来下载文件(这种方式通用度比较高);
    这种方式通用度比较高,不好的是前端要先把后端返回的文件流全部获取,才能进行下载。在获取文件流这段时间是前端ajax异步加载过程。

这2种方式都是把文件流存在浏览器内存中,比较依赖于客户端硬件配置。手机端下载大文件一般会直接写到硬盘(手机存储)中。

2.2.1 直接下载-利用a标签
downloadFileByAtagClick(url, fileName = '') {
      let a = document.createElement('a');
      a.style.display = 'none';
      a.href = url;
      document.body.appendChild(a);
      fileName && a.setAttribute('download', fileName);
      a.click();
      document.body.removeChild(a);

      // 方法二: saveAs(url);
    },

这个可以直接利用file-saversaveAs方法,它会根据传入的参数,来选择使用blob还是a [href]

2.2.2 直接下载-利用iframe标签
downloadFileByIframe(url) {
      let iframe = document.createElement('iframe');
      iframe.style.display = 'none';
      iframe.src = url;
      document.body.appendChild(iframe);
      setTimeout(() => document.body.removeChild(iframe), 100);
    },

iframe [src]标签跟a [href]标签原理差不多。

2.2.3 blob-原生js
downloadFileByBlob(url, method = 'get') {
      // 使用这种方式多了创建请求的时间
      let fileName = url.split('fileName=')[1];
      fetch(url, {
        method
      }).then(res => {
        console.log(res.headers.get('content-length'));
        return res.blob();
      }).then(data => {
          let blob = new Blob([data], {
            type: 'application/octet-stream'
          });
          // url表示指定的 File 对象或 Blob 对象。
          let url = URL.createObjectURL(blob);
          this.downloadFileByAtagClick(url, fileName);
          // 释放URL对象
          URL.revokeObjectURL(url);
        })
        .catch(error => console.error('Error:', error));
    },
2.2.3 blob-"file-saver"saveAs方法
downloadFileByFileSaver(url, method) {
      let fileName = url.split('fileName=')[1];
      fetch(url, {
        method
      }).then(res => res.blob())
        .then(data => {
          let blob = new Blob([data], {
            type: 'application/octet-stream'
          });
          saveAs(blob, fileName);
        })
        .catch(error => console.error('Error:', error));
    },

3、后端部分

后端主要依赖库:fs、express、multer、elasticsearch,除了fs,其它模块请提前npm install。

3.1 后端代码-web交互

/**用来实现文件上传、下载、删除功能**/
let fs = require('fs');
let express = require('express');
let multer = require('multer');
let esUtil = require('./esUtil');

let app = express();
let storage = multer.diskStorage({
  destination: 'file/',
  filename: function (req, file, cb) {
    cb(null, file.originalname);
  }
});
let upload = multer({ storage: storage });
esUtil.initEsUtil({ host: 'localhost:9200' });
app.listen(3100);

// 单文件上传,file指代上传时候的文件属性名
app.post('/file', upload.single('file'), async (req, res, next) => {
  let file = req.file;
  let fileInfo = {};

  // 获取文件信息
  fileInfo.mimetype = file.mimetype;
  fileInfo.fileName = file.originalname;
  fileInfo.size = file.size;
  fileInfo.path = file.path;
  fileInfo.dateTime = new Date();

  // 存入es
  let result = await esUtil.post('files', 'file', fileInfo);
  res.send(result);
});

// 文件下载
app.get('/file', (req, res, next) => {
  let fileName = req.query.fileName;
  let path = './file/' + fileName;
  let size = fs.statSync(path).size; // 文件大小
  res.writeHead(200, {
    // 告诉浏览器文件是二进制文件,不想直接显示内容
    'Content-type': 'application/octet-stream',
    // 告诉浏览器这是一个需要下载的文件(以附件的形式下载),设置下载文件名
    'Content-Disposition': 'attachment; filename=' + encodeURI(fileName),
    'Content-Length': size
  });
  let readStream = fs.createReadStream(path); // 得到输入文件流
  readStream.pipe(res);
});

// 文件删除
app.delete('/file', async (req, res, next) => {
  let fileName = req.query.fileName;
  let results = await esUtil.get('files', 'file', {
    term: {
      'fileName.keyword': {
        value: fileName
      }
    }
  });
  // 删除指定数据
  let result = await esUtil.delete('files', 'file', req.query.id);
  if (results.data.length === 1 && result.status === 200) fs.unlinkSync('./file/' + fileName);
  res.send(result);
});

// 获取文件列表
app.get('/fileList', async (req, res, next) => {
  let results = await esUtil.getAll('files', 'file');
  res.send(results);
});

app.use((req, res, next, err) => {
  res.send({status: 500});
});
3.1.1 文件上传说明

multer实现文件的上传依赖库,可配置文件上传路径、单个文件上传和多个文件上传等。本文配置的单文件上传,上传目录是当前执行代码相对路径file/

3.1.2 文件下载说明

后端返回响应头需要添加:

  • 'Content-type': 'application/octet-stream',告诉浏览器文件是二进制文件,不想直接显示内容。
  • 'Content-Disposition': 'attachment; filename=' + encodeURI(fileName),告诉浏览器这是一个需要下载的文件(以附件的形式下载),并设置下载文件名。
  • 'Content-Length': size,设置文件大小,这样前端在直接下载文件的时候会显示文件大小,对用户交互界面也更加友好。

3.2 后端代码-es操作

let elasticsearch = require('elasticsearch');
let client;
const ERROR_STATUS = 500;
const SUCCESS_STATUS = 200;

exports.initEsUtil = (config) => {
  client = elasticsearch.Client(config);
  return client;
};

// 插入一条数据
exports.post = (index, type, data) => {
  return new Promise((resolve, reject) => {
    client.index({
      index,
      type,
      body: data
    }, (err, res) => {
      err ? reject({status: ERROR_STATUS}) : resolve({status: SUCCESS_STATUS});
    });
  });
};

// 获取所有数据
exports.getAll = async (index, type) => {
  let res = await client.search({
    index,
    type,
    body: {
      sort: [
        {
          dateTime: {
            order: 'desc'
          }
        }
      ],
    }
  });

  let list = [];
  if (res instanceof Error) return {status: ERROR_STATUS, data: list};
  if (res && res.hits && res.hits.hits) {
    let arr = res.hits.hits;
    for (let item of arr) {
      list.push(Object.assign({id: item._id}, item._source));
    }
  }

  return {status: SUCCESS_STATUS, data: list};
};

// 查询指定数据
exports.get = (index, type, param) => {
  return new Promise((resolve, reject) => {
    client.search({
      index,
      type,
      body: {
        query: param
      }
    }, (err, res) => {
      let list = [];
      if (res && res.hits && res.hits.hits) {
        let arr = res.hits.hits;
        for (let item of arr) {
          list.push(Object.assign({id: item._id}, item._source));
        }
      }
      err ? reject({status: ERROR_STATUS, data: []}) : resolve({status: SUCCESS_STATUS, data: list});
    });
  });
};

// 删除指定数据
exports.delete = (index, type, id) => {
  return new Promise((resolve, reject) => {
    client.delete({
      index,
      type,
      id
    }, (err, res) => {
      err ? reject({status: ERROR_STATUS}) : resolve({status: SUCCESS_STATUS});
    });
  });
};
3.2.1 es操作说明

es操作还是新手级别,只会用很简单的操作。不过最好把es相关操作和web交互代码分开,看着舒服一点。代码种用了async awite promise相关写法,代码看起来直观一点,也避免回调地狱。

4、遗留问题

  • 超大文件的上传和下载,如何友好处理;
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,402评论 6 499
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,377评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,483评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,165评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,176评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,146评论 1 297
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,032评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,896评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,311评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,536评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,696评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,413评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,008评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,659评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,815评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,698评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,592评论 2 353

推荐阅读更多精彩内容