NodeJs之文件上传

一,介绍与需求

1.1,介绍

1,multer模块

multer用于处理文件上传的nodejs中间件,主要跟express框架搭配使用,只支持表单MIME编码为multipart/form-data类型的数据请求.

2,fs模块

fs模块用于对系统文件及目录进行读写操作。

1.2,需求

上传并操作文件与文件目录

二,配置实现

2.1,multer模块

第一步:安装multer模块

 cnpm install multer --save

第二步:引入multer模块,构造multer对象: multer(opt)

opt是个key-value对象,包含属性dest/storage,fileFilter,limits.分表表示文件的存储位置/方式,文件过滤,文件大小限制.如下:

 var moment = require("moment");
var express = require("express");
var multer = require('multer');
var storage = multer.diskStorage({
//文件存储路径
destination: function (req, file, cb) {
cb(null, path.join(__dirname, "/../uploads/temps"));
},
//修改上传文件的名字
//file 是个文件对象 ,fieldname对应在客户端的name属性
//存储的文件需要自己加上文件的后缀,multer并不会自动添加
//这里直接忽略文件的后缀.
filename: function (req, file, cb) {
var date = new Date();
cb(null, moment().format("YYYYMMDDhhmmss") + file.originalname);
}
});
let objMulter = multer({storage : storage });

如果初始化multer时候没有指定dest或者storage,上传的文件将保存在内存中,永远不会写入到磁盘中,在storage中如果没有指定destination值,那么上传的文件将存储在系统默认的临时文件夹.

 multer关联的文件信息:

 filedname   : 在form表单中指定的name属性值
orginalname : 原始文件名
encoding : 文件编码方式
mimetype : 多媒体类型
size : 文件大小,单位b
destination : 文件上传后存储在服务端的路径
filename : 文件在服务端的命名
path : 文件在服务端的完整路径
buffer : 文件二进制数据 ps:其中destination,filename和path只有在指定storage属性时候有效.而buffer只有文件在内存中存储时候有效.

multer同时提供了single/array/fields/any方法用于对接受文件数的控制.

  1. single(fieldname) 接收单个文件,通过req.file访问该文件
  2. array(fieldnaem,[maxcount]) 接收多个文件,通过req.files数组方法文件.maxcount指定接收文件最大数.
  3. fields(fields) 接受指定fieldname的文件,fieldname由客户端决定,通过req.files数组方法
  4. any 接收所有文件上传,通过req.files访问文件

2.2,fs模块

第一步:安装fs模块,也可直接引用,node默认集成

 cnpm install fs --save//项目目录安装
var fs = require("fs");//直接引入文件系统模块

第二步:介绍fs模块的方法

 var fs = require('fs'); // 载入fs模块

1,删除指定文件:fs.unlink(path,callback)

path参数为该文件的绝对物理路径,callback回调参数当中只有一个错误信息参数err,一般在该文件不存在或者删除文件失败时触发调用。

 fs.unlink('/tmp/234.txt', function(err) {
if (err) {
throw err;
}
console.log('成功删除了 /tmp/234.txt');
});

2,读取文件:fs.readFile(filename,[option],callback)

参数说明:

    • filename String 文件名
    • option Object
      • encoding String |null default=null
      • flag String default='r'
    • callback Function

path参数为该文件的绝对物理路径,其中options参数可选,可以传入编码格式,如读取文本文件时,可传入'utf8',若不指定编码格式,则默认输出读取的文件内容为Buffer形式,故一般都会传入该参数。callback回调参数当中有两个参数errdata,其中err为错误信息参数,一般在在文件不存在或者读取文件失败时触发调用,data为文件内容。

 fs.readFile('./tmp/test.txt','utf-8'function(err, data) { 
// 读取文件失败/错误
if (err) {
throw err;
}
// 读取文件成功
console.log(data);
});

3,写入文件:fs.writeFile(filename,data,[options],callback),追加写入:fs.appendFile(filename,data[,options],callback)

filename参数为该文件的绝对物理路径,data为需要写入该文件当中的数据内容,其中options参数可选,可以传入编码格式,若不传则默认为utf8callback回调参数当中只有一个错误信息参数err,一般在写入失败时触发调用。

 // 写入文件内容(如果文件不存在会创建一个文件)
// 传递了追加参数 { 'flag': 'a' }
fs.writeFile('./test2.txt', 'test test', { 'flag': 'a' }, function(err) {
if (err) {
throw err;
} console.log('Saved.'); // 写入成功后读取测试
fs.readFile('./test2.txt', 'utf-8', function(err, data) {
if (err) {
throw err;
}
console.log(data);
});
});

flag传值,r代表读取文件,w代表写文件,a代表追加。

 // 追加写入
fs.appendFile("2.txt","我是追加的字符",function (err) {
if(err){
return console.log(err);
}else {
console.log("追加成功");
}
})

4,检测文件是否存在:fs.exists(path,callback)

path参数传入该文件的绝对物理路径,该callback回调函数有个参数exists。exists为一个文件状态对象,是否存在。

 var dest_Dir = path.join(__dirname.replace("routes", ""), "uploads", destDir);
fs.exists(dest_Dir, function (exists) {
if (exists) {
//存在
}
else { }
});

5,移动或重命名指定文件:fs.rename(oldPath,newPath,callback)

oldPath参数为该文件原来的路径,newPath参数为该文件移动或重命名之后的路径,这两个参数都必须能传入文件完整的绝对物理路径。callback回调参数当中只有一个错误信息参数,一般在oldPath当中指定的文件不存在或者该操作失败时触发调用。

  fs.rename(sourceFile, destPath, function (err) {
res.status(200).json({
httpCode: 200,
message: '上传成功',
data: {
"fileurl": fileurl
},
});
});

6,创建一个目录文件夹:fs.mkdir(path[,model],callback)

path为该目录的绝对物理路径,callback回调函数当中也只有一个错误信息参数,一般在目录创建失败时触发调用。

  fs.mkdir(dest_Dir, 0777, function (err) {
if (err) {
res.status(500).json({ } else {
//创建目录成功,上传文件
fs.rename(sourceFile, destPath, function (err) {
res.status(200).json({
httpCode: 200,
message: '上传成功',
data: {
"fileurl": fileurl
},
});
});
}
})

7,读取目录文件夹:fs.readdir(path,callback)

path为该目录的绝对物理路径,callback回调函数当中有两个参数errfileserr为错误信息参数,一般在该目录不存在或读取失败时触发调用,files为一个数组对象,包含该目录下的所有文件夹与文件的名字。(仅为文件夹的名字和文件名,不是路径形式)。

 fs.readdir('./newdir', function(err, files) {
if (err) {
throw err;
}
// files是一个数组
// 每个元素是此目录下的文件或文件夹的名称
console.log(files);
});

8,删除一个空文件夹:fs.rmdir(path,callback)

path为该目录的绝对物理路径,callback回调函数当中也只有一个错误信息参数,一般在该目录不存在或者删除操作失败时触发调用。

 fs.rmdir('/tmp', function(err) {
if (err) {
throw err;
}
console.log('成功删除了 空文件夹 tmp');
});

以上的实例均是异步操作,如果需同步操作在函数后面加Sync既可

 //同步 所有同步的函数都是函数后面加Sync;
var res = fs.writeFileSync("1.txt","我是写入内容");

2.3,文件上传

1,单文件上传

 router.post('/upload', upload.single('file'), function (req, res, next) {
var fileName = req.file.filename;
var destDir = req.body.dir == undefined ? "default" : req.body.dir;
var sourceFile = req.file.path;
var destPath = path.join(__dirname.replace("routes", ""), "uploads", destDir, fileName);
var dest_Dir = path.join(__dirname.replace("routes", ""), "uploads", destDir);
var fileurl = uploadFileDomin + destPath.substr(destPath.indexOf("uploads"));
fileurl = fileurl.replace(/\\/g, "/");
fs.exists(dest_Dir, function (exists) {
if (exists) {
fs.rename(sourceFile, destPath, function (err) {
res.status(200).json({
httpCode: 200,
message: '上传成功',
data: {
"fileurl": fileurl
},
});
});
}
else {
fs.mkdir(dest_Dir, 0777, function (err) {
if (err) {
res.status(500).json({
httpCode: 500,
message: err,
data: [],
});
} else {
fs.rename(sourceFile, destPath, function (err) {
res.status(200).json({
httpCode: 200,
message: '上传成功',
data: {
"fileurl": fileurl
},
});
});
}
})
}
}); });

2,多文件上传

 router.post('/uploads', upload.array('file', 8), function (req, res, next) {
var paths = [];
for (var i = 0; i < req.files.length; i++) {
var path = req.files[i].path.replace(/\\/g, "/");
var fileurl = uploadFileDomin + path.substr(path.indexOf("uploads")).replace('\\', '/');
paths.push(fileurl);
}
res.status(200).json({
httpCode: 200,
message: '上传成功',
"fileurls": paths,
});
});

最后抛出发送上传接口:

 app.use("/api", require("./fileUploadApi.js"));

三,前端调用

3.1,单文件上传调用

1,简单html

 <span>ajax upload</span>
<div class="container">
<label>file</label>
<input type="file" name="file" id="file">
<input type='button' value='上传' id="btn_upload" />
</div>

2,js调用

  $("#btn_upload").click(function () {
//创建一个FormDate
var formData = new FormData();
//将文件信息追加到其中
formData.append('file', file.files[0]);
formData.append('dir', 'attachment');
//formData.append('name', file.files[0].name);
$.ajax({
url: "http://localhost:3000/api/" + 'upload',
type: 'POST',
data: formData,
cache: false,
contentType: false,
processData: false,
xhrFields: {
withCredentials: true
},
crossDomain: true,
success: function (data) {
callBack(data);
},
error: function (response) {
console.log("error is :" + response);
}
});

3,实现效果

3.2,多文件上传调用

1,简单html

   <span>ajax multi files upload</span>
<div class="container">
<label>file</label>
<input type="file" name="file" id="files1">
<input type="file" name="file" id="files2">
<input type="file" name="file" id="files3">
<input type='button' value='上传' id="btn_uploads" />
</div>

2,js调用

         var files1 = $("#files1")[0];
var files2 = $("#files2")[0];
var files3 = $("#files3")[0];
$("#btn_uploads").click(function () {
//创建一个FormDate
var formData = new FormData();
//将文件信息追加到其中
formData.append('file', files1.files[0]);
formData.append('file', files2.files[0]);
formData.append('file', files3.files[0]);
$.ajax({
url: 'http://localhost:3000/api/' + 'uploads',
type: 'POST',
data: formData,
contentType: false,
processData: false,
async: false,
xhrFields: {
withCredentials: true
},
crossDomain: true,
success: function (data) {
callBack(data.fileurls);
},
error: function (response) {
console.log("error is :" + response);
}
})
})

3,实现效果

NodeJs之文件上传的更多相关文章

  1. Nodejs express 文件上传

    文件上传 以下我们创建一个用于上传文件的表单,使用 POST 方法,表单 enctype 属性设置为 multipart/form-data. index.htm 文件代码修改如下: <html ...

  2. nodejs实现文件上传

    在使用ant-design的upload上传文件时,前端很好实现,那么我们如何实现node服务端呢? 服务端文件上传实现 var express = require('express'); var f ...

  3. nodejs+multiparty 文件上传

    通过表单提交上传文件:     html代码 <form action="/uploadFile" method="post" enctype=" ...

  4. nodeJs实现文件上传,下载,删除

    转:https://blog.csdn.net/qq_36228442/article/details/81709272 一.简介 本文介绍了nodeJs+express框架下,用multer中间件实 ...

  5. nodejs应用:文件上传

    功能:上传文件到服务器,图片支持客户端本地预览. 服务端 //server.js 'use strict';const http = require('http');const url = requi ...

  6. 【nodejs】文件上传demo实现

    文件结构: index.js var server = require('./server.js'); var router = require('./router.js'); var request ...

  7. Html5+NodeJS——拖拽多个文件上传到服务器

    实现多文件拖拽上传的简易Node项目,可以在github上下载,你可以先下载下来:https://github.com/Johnharvy/upLoadFiles/. 解开下载下的zip格式包,建议用 ...

  8. Nodejs进阶:基于express+multer的文件上传

    关于作者 程序猿小卡,前腾讯IMWEB团队成员,阿里云栖社区专家博主.欢迎加入 Express前端交流群(197339705). 正在填坑:<Nodejs学习笔记> / <Expre ...

  9. nodejs学习之文件上传

    最近要做个图片上传的需求,因为服务端春节请假回家还没来,所以就我自己先折腾了一下,大概做出来个效果,后台就用了nodejs,刚开始做的时候想网上找一下资料,发现大部分资料都是用node-formida ...

随机推荐

  1. Python 闭包小记

    闭包就是能够读取其他函数内部变量的函数.例如在javascript中,只有函数内部的子函数才能读取局部变量,所以闭包可以理解成“定义在一个函数内部的函数“.在本质上,闭包是将函数内部和函数外部连接起来 ...

  2. java~springboot~ibatis数组in查询的实现

    在ibatis的xml文件里,我们去写sql语句,对应mapper类的方法,这些sql语句与控制台上没什么两样,但在有些功能上需要注意,如where in这种从数组里查询符合条件的集合里,需要在xml ...

  3. MySQL系列--4.使用Python3访问数据库

    1.安装MySQL驱动 pip install mysql-connector 安装完成后进入命令行模式,导入驱动,如果不报错,说明安装成功 Python 3.6.7 (default, Oct 22 ...

  4. 深入Go的底层,带你走近一群有追求的人

    目录 缘起 自我介绍的技巧 硬核知识点 什么是plan9汇编 汇编角度看函数调用及返回过程 汇编角度看slice 正确参与Go夜读活动的方式 阅读原文 上周六晚上,我参加了"Go夜读&quo ...

  5. [翻译] GCC 内联汇编 HOWTO

    目录 GCC 内联汇编 HOWTO 原文链接与说明 1. 简介 1.1 版权许可 1.2 反馈校正 1.3 致谢 2. 概览 3. GCC 汇编语法 4. 基本内联 5. 扩展汇编 5.1 汇编程序模 ...

  6. 记一次按需加载和npm模块发布实践

    按需加载 在使用 lodash 的时候我们可以使用这样的代码 //一 import {omit} from "lodash"; //二 import l from "lo ...

  7. [Vue] vue中setInterval的问题

    vue中使用setInterval this.chatTimer = setInterval(() => { console.log(this.chatTimer); this.chatMsg( ...

  8. sql 修改、更新、替换 某个字段的部分内容(转载)

    来源:https://blog.csdn.net/jiangnanqbey/article/details/81304834 1*.需求 将表(Ws_FormMain)的字段(order_Number ...

  9. JS 的继承

    1:原生链:prototype 儿子能够继承父亲的属性,也可以觉得遗传基因不好自己改属性,(但是不能改变老爸的属性). 看例子:             function farther(){     ...

  10. 一看就能学会的H5视频推流方案

    本文由云+社区发表 作者:周超 导语 随着直播平台爆发式增长,直播平台从 PC 端转战移动端,紧跟着直播的潮流,自己学习实现了一套简单的 H5 视频推流的解决方案,下面就给小伙伴们分享一下自己学习过程 ...