本篇文章給大家分享的是有關怎么在node中對文件上傳接口進行轉發(fā),小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。
創(chuàng)新互聯公司是一家專注于成都網站建設、網站設計與策劃設計,大寧網站建設哪家好?創(chuàng)新互聯公司做網站,專注于網站建設十載,網設計領域的專業(yè)建站公司;建站業(yè)務涵蓋:大寧等地區(qū)。大寧做網站價格咨詢:028-86922220
js 中的文件
web 中的 Blob 、File 和 Formdate
一個 Blob ( Binary Large Object ) 對象表示一個不可變的, 原始數據的類似文件對象。Blob表示的數據不一定是一個JavaScript原生格式。 File 接口基于Blob,繼承 Blob 功能并將其擴展為支持用戶系統上的文件。
前端上傳文件的方式無非就是使用:1、表單自動上傳;2、使用 ajax 上傳。我們可以使用以下代碼創(chuàng)建一個 Form,并打印出 file
從 F12 中可以看出 File 原型鏈上是 Blob。
簡單地說 Blob 可以理解為 Web 中的二進制文件。 而 File 是基于 Blob 實現的一個類,新增了關于文件有關的一些信息。
FormData對象的作用就類似于 Jq 的 serialize() 方法,不過 FormData 是瀏覽器原生的,且支持二進制文件。 ajax 通過 FormData 這個對象發(fā)送表單請求,無論是原生的 XMLHttpRequest 、jq 的 ajax 方法、 axios 都是在 data 里直接指定上傳 formData 類型的數據,fetch api 是在 body 里上傳。
forData 數據有兩種方式生成,如下 formData 和 formData2 的區(qū)別,而 formData2 可以通過傳入一個 element 的方式進行初始化,初始化之后依然可以調用 formData 的 append 方法。
console.log() 無法直接打印出 formData 的數據,可以使用 get(key) 或者 getAll(key)
如果是使用 new FormData(element) 的創(chuàng)建方式,上面 key 為 上的 name 字段。
如果是使用 append 添加的數據,get/getAll 時 key 為 append 所指定的 key。
node 中的 Buffer 、 Stream 、fs
Buffer 和 Stream 是 node 為了讓 js 在后端擁有處理二進制文件而出現的數據結構。
通過名字可以看出 buffer 是緩存的意思。存儲在內存當中,所以大小有限,buffer 是 C++ 層面分配的,所得內存不在 V8 內。
stream 可以用水流形容數據的流動,在文件 I/O、網絡 I/O中數據的傳輸都可以稱之為流。
通過兩個 fs 的 api 看出,readFile 不指定字符編碼默認返回 buffer 類型,而 createReadStream 將文件轉化為一個 stream , nodejs 中的 stream 通過 data 事件能夠一點一點地拿到文件內容,直到 end 事件響應為止。
const fs = require("fs"); fs.readFile("./package.json", function(err, buffer) { if (err) throw err; console.log("buffer", buffer); }); function readLines(input, func) { var remaining = ""; input.on("data", function(data) { remaining += data; var index = remaining.indexOf("\n"); var last = 0; while (index > -1) { var line = remaining.substring(last, index); last = index + 1; func(line); index = remaining.indexOf("\n", last); } remaining = remaining.substring(last); }); input.on("end", function() { if (remaining.length > 0) { func(remaining); } }); } function func(data) { console.log("Line: " + data); } var input = fs.createReadStream("./package.json"); input.setEncoding("binary"); readLines(input, func);
fs.readFile() 函數會緩沖整個文件。 為了最小化內存成本,盡可能通過 fs.createReadStream() 進行流式傳輸。
使用 nodejs 創(chuàng)建 uoload api
http 協議中的文件上傳
在 http 的請求頭中 Content-type 是 multipart/form-data時,請求的內容如下:
POST / HTTP/1.1 Content-Type: multipart/form-data; boundary=----WebKitFormBoundaryoMwe4OxVN0Iuf1S4 Origin: http://localhost:3000 Referer: http://localhost:3000/upload Sec-Fetch-Mode: navigate Sec-Fetch-User: ?1 Upgrade-Insecure-Requests: 1 User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/76.0.3809.132 Safari/537.36 ------WebKitFormBoundaryoqBx9oYBhx4SF1YQ Content-Disposition: form-data; name="upload" http://localhost:3000 ------WebKitFormBoundaryoMwe4OxVN0Iuf1S4 Content-Disposition: form-data; name="upload"; filename="IMG_9429.JPG" Content-Type: image/jpeg ????JFIF??C // 文件的二進制數據 …… --------WebKitFormBoundaryoMwe4OxVN0Iuf1S4--
根據 WebKitFormBoundaryoMwe4OxVN0Iuf1S4 可以分割出文件的二進制內容
原生 node
使用原生的 node 寫一個文件上傳的 demo
const http = require("http"); const fs = require("fs"); const util = require("util"); const querystring = require("querystring"); //用http模塊創(chuàng)建一個http服務端 http .createServer(function(req, res) { if (req.url == "/upload" && req.method.toLowerCase() === "get") { //顯示一個用于文件上傳的form res.writeHead(200, { "content-type": "text/html" }); res.end( '" ); } else if (req.url == "/upload" && req.method.toLowerCase() === "post") { if (req.headers["content-type"].indexOf("multipart/form-data") !== -1) parseFile(req, res); } else { res.end("pelease upload img"); } }) .listen(3000); function parseFile(req, res) { req.setEncoding("binary"); let body = ""; // 文件數據 let fileName = ""; // 文件名 // 邊界字符串 ----WebKitFormBoundaryoMwe4OxVN0Iuf1S4 const boundary = req.headers["content-type"] .split("; ")[1] .replace("boundary=", ""); req.on("data", function(chunk) { body += chunk; }); req.on("end", function() { const file = querystring.parse(body, "\r\n", ":"); // 只處理圖片文件; if (file["Content-Type"].indexOf("image") !== -1) { //獲取文件名 var fileInfo = file["Content-Disposition"].split("; "); for (value in fileInfo) { if (fileInfo[value].indexOf("filename=") != -1) { fileName = fileInfo[value].substring(10, fileInfo[value].length - 1); if (fileName.indexOf("\\") != -1) { fileName = fileName.substring(fileName.lastIndexOf("\\") + 1); } console.log("文件名: " + fileName); } } // 獲取圖片類型(如:image/gif 或 image/png)) const entireData = body.toString(); const contentTypeRegex = /Content-Type: image\/.*/; contentType = file["Content-Type"].substring(1); //獲取文件二進制數據開始位置,即contentType的結尾 const upperBoundary = entireData.indexOf(contentType) + contentType.length; const shorterData = entireData.substring(upperBoundary); // 替換開始位置的空格 const binaryDataAlmost = shorterData .replace(/^\s\s*/, "") .replace(/\s\s*$/, ""); // 去除數據末尾的額外數據,即: "--"+ boundary + "--" const binaryData = binaryDataAlmost.substring( 0, binaryDataAlmost.indexOf("--" + boundary + "--") ); // console.log("binaryData", binaryData); const bufferData = new Buffer.from(binaryData, "binary"); console.log("bufferData", bufferData); // fs.writeFile(fileName, binaryData, "binary", function(err) { // res.end("sucess"); // }); fs.writeFile(fileName, bufferData, function(err) { res.end("sucess"); }); } else { res.end("reupload"); } }); }
通過 req.setEncoding("binary"); 拿到圖片的二進制數據。可以通過以下兩種方式處理二進制數據,寫入文件。
fs.writeFile(fileName, binaryData, "binary", function(err) { res.end("sucess"); });
fs.writeFile(fileName, bufferData, function(err) { res.end("sucess"); });
koa
在 koa 中使用 koa-body 可以通過 ctx.request.files 拿到上傳的 file 對象。下面是例子。
'use strict'; const Koa = require('koa'); const app = new Koa(); const router = require('koa-router')(); const koaBody = require('../index')({multipart:true}); router.post('/users', koaBody, (ctx) => { console.log(ctx.request.body); // => POST body ctx.body = JSON.stringify(ctx.request.body, null, 2); } ); router.get('/', (ctx) => { ctx.set('Content-Type', 'text/html'); ctx.body = `