问答题1051/1593如何实现文件上传?说说你的思路

难度:
2021-07-25 创建

参考答案:

一、是什么

文件上传在日常开发中应用很广泛,我们发微博、发微信朋友圈都会用到了图片上传功能

因为浏览器限制,浏览器不能直接操作文件系统的,需要通过浏览器所暴露出来的统一接口,由用户主动授权发起来访问文件动作,然后读取文件内容进指定内存里,最后执行提交请求操作,将内存里的文件内容数据上传到服务端,服务端解析前端传来的数据信息后存入文件里

对于文件上传,我们需要设置请求头为content-type:multipart/form-data

multipart互联网上的混合资源,就是资源由多种元素组成,form-data表示可以使用HTML Forms 和 POST 方法上传文件

结构如下:

1POST /t2/upload.do HTTP/1.1 2User-Agent: SOHUWapRebot 3Accept-Language: zh-cn,zh;q=0.5 4Accept-Charset: GBK,utf-8;q=0.7,*;q=0.7 5Connection: keep-alive 6Content-Length: 60408 7Content-Type:multipart/form-data; boundary=ZnGpDtePMx0KrHh_G0X99Yef9r8JZsRJSXC 8Host: w.sohu.com 9 10--ZnGpDtePMx0KrHh_G0X99Yef9r8JZsRJSXC 11Content-Disposition: form-data; name="city" 12 13Santa colo 14--ZnGpDtePMx0KrHh_G0X99Yef9r8JZsRJSXC 15Content-Disposition: form-data;name="desc" 16Content-Type: text/plain; charset=UTF-8 17Content-Transfer-Encoding: 8bit 18 19... 20--ZnGpDtePMx0KrHh_G0X99Yef9r8JZsRJSXC 21Content-Disposition: form-data;name="pic"; filename="photo.jpg" 22Content-Type: application/octet-stream 23Content-Transfer-Encoding: binary 24 25... binary data of the jpg ... 26--ZnGpDtePMx0KrHh_G0X99Yef9r8JZsRJSXC--

boundary表示分隔符,如果要上传多个表单项,就要使用boundary分割,每个表单项由———XXX开始,以———XXX结尾

xxx是即时生成的字符串,用以确保整个分隔符不会在文件或表单项的内容中出现

每个表单项必须包含一个 Content-Disposition 头,其他的头信息则为可选项, 比如 Content-Type

Content-Disposition 包含了 type 和 一个名字为nameparametertypeform-dataname 参数的值则为表单控件(也即 field)的名字,如果是文件,那么还有一个 filename 参数,值就是文件名

1Content-Disposition: form-data; name="user"; filename="logo.png"

至于使用multipart/form-data,是因为文件是以二进制的形式存在,其作用是专门用于传输大型二进制数据,效率高

二、如何实现

关于文件的上传的上传,我们可以分成两步骤:

  • 文件的上传
  • 文件的解析

文件上传

传统前端文件上传的表单结构如下:

1<form action="http://localhost:8080/api/upload" method="post" enctype="multipart/form-data"> 2 <input type="file" name="file" id="file" value="" multiple="multiple" /> 3 <input type="submit" value="提交"/> 4</form>

action 就是我们的提交到的接口,enctype="multipart/form-data" 就是指定上传文件格式,inputname 属性一定要等于file

文件解析

在服务器中,这里采用koa2中间件的形式解析上传的文件数据,分别有下面两种形式:

  • koa-body
  • koa-multer

koa-body

安装依赖

1npm install koa-body

引入koa-body中间件

1const koaBody = require('koa-body'); 2app.use(koaBody({ 3 multipart: true, 4 formidable: { 5 maxFileSize: 200*1024*1024 // 设置上传文件大小最大限制,默认2M 6 } 7}));

获取上传的文件

1const file = ctx.request.files.file; // 获取上传文件

获取文件数据后,可以通过fs模块将文件保存到指定目录

1router.post('/uploadfile', async (ctx, next) => { 2 // 上传单个文件 3 const file = ctx.request.files.file; // 获取上传文件 4 // 创建可读流 5 const reader = fs.createReadStream(file.path); 6 let filePath = path.join(__dirname, 'public/upload/') + `/${file.name}`; 7 // 创建可写流 8 const upStream = fs.createWriteStream(filePath); 9 // 可读流通过管道写入可写流 10 reader.pipe(upStream); 11 return ctx.body = "上传成功!"; 12});

koa-multer

安装依赖:

1npm install koa-multer

使用 multer 中间件实现文件上传

1const storage = multer.diskStorage({ 2 destination: (req, file, cb) => { 3 cb(null, "./upload/") 4 }, 5 filename: (req, file, cb) => { 6 cb(null, Date.now() + path.extname(file.originalname)) 7 } 8}) 9 10const upload = multer({ storage}); 11const fileRouter = new Router(); 12fileRouter.post("/upload", upload.single('file'), (ctx, next) => { 13 console.log(ctx.req.file); // 获取文件 14}) 15 16app.use(fileRouter.routes());

最近更新时间:2024-08-10

赞赏支持

预览

题库维护不易,您的支持就是我们最大的动力!