Ajax異步文件上傳與NodeJS express服務端處理
為了避免在實現(xiàn)簡單的異步文件上傳功能時候引入一個第三方庫文件的尷尬情形(庫文件可能造成多余的開銷,拉低應用加載速度,尤其是在引入庫文件之后僅使用其中一兩個功能的情況下,性價比極低),最近了解了一下文件異步上傳的實現(xiàn)原理,順帶看了看進度條、圖片預覽等功能的實現(xiàn),做一點簡單的整理。
文件上傳
HTML結構如下,一個file input和一個button。當點擊“上傳”按鈕的時候,將file input選中的文件上傳到服務器。
<input type="file" name="file" id="file" /> <button id="upload">上傳</button>
以下是“上傳”按鈕的點擊事件處理器,點擊按鈕之后通過一個XMLHttpRequest對象來實現(xiàn)發(fā)送異步請求。上傳的內(nèi)容為文件,因此還需要用到FormData對象,F(xiàn)ormData可以js里面創(chuàng)建表單對象,將file input的文件append到FormData對象中,最后調(diào)用XHR對象的send()方法將表單數(shù)據(jù)發(fā)送出去即可。
var file = document.querySelector('#file'); var upload = document.querySelector('#upload'); var xhr = new XMLHttpRequest(); // 點擊上傳 function uploadFile(event) { var formData = new FormData(); formData.append('test-upload', file.files[0]); xhr.onload = uploadSuccess; xhr.open('post', '/upload', true); xhr.send(formData); } // 成功上傳 function uploadSuccess(event) { if (xhr.readyState === 4) { console.log(xhr.responseText); } }
上傳進度
在進行文件上傳的時候,xhr對象會有一個upload屬性,會提供一個progress事件,在相應的事件處理器里面通過事件對象可以知道當前的上傳進度,利用這個特點可以很方便地實現(xiàn)進度條或者進度提示。
<input type="file" name="file" id="file" /> <button id="upload">上傳</button> <span id="progress">0%</span>
var progress = document.querySelector('#progress'); // 點擊上傳 function uploadFile(event) { var formData = new FormData(); formData.append('test-upload', file.files[0]); xhr.onload = uploadSuccess; xhr.upload.onprogress = setProgress; xhr.open('post', '/upload', true); xhr.send(formData); } // 進度條 function setProgress(event) { if (event.lengthComputable) { var complete = Number.parseInt(event.loaded / event.total * 100); progress.innerHTML = complete + '%'; } }
圖片預覽
上傳圖片的時候可以利用FileReader對象來實現(xiàn)圖片預覽。FileReader可以異步讀取用戶電腦上的文件,將file input選中的文件傳給FileReader,讀取之后取得文件的URL并設置為image元素的src即可讓選中的圖片文件顯示出來。
<input type="file" name="file" id="file" /> <button id="upload">上傳</button> <span id="progress">0</span> <img id="image" src="" width="200" />
var file = document.querySelector('#file'); file.addEventListener('change', previewImage, false); // 圖片預覽 function previewImage(event) { var reader = new FileReader(); reader.onload = function (event) { image.src = event.target.result; }; reader.readAsDataURL(event.target.files[0]); }
服務端處理
使用express搭建一個簡單的NodeJS服務端,提供上傳文件的接口。express要支持文件上傳需要用到中間件,在express官網(wǎng)上有很多介紹。這里我使用的是multer中間件,下面是簡單的使用示例。upload.single表示這個接口接受的上傳文件數(shù)量為1個,'test-upload'限制了上傳的表單數(shù)據(jù)的鍵為'test-upload'(formData.append(‘test-upload', file.files[0]);)。經(jīng)過這個中間件處理之后,通過req.file可以訪問到文件的相關信息,上傳的文件存放在uploads文件夾中。
const upload = require('multer')({ dest: 'uploads/' }); app.post('/upload', upload.single('test-upload'), (req, res) => { // 沒有附帶文件 if (!req.file) { res.json({ ok: false }); return; } // 輸出文件信息 console.log('===================================================='); console.log('fieldname: ' + req.file.fieldname); console.log('originalname: ' + req.file.originalname); console.log('encoding: ' + req.file.encoding); console.log('mimetype: ' + req.file.mimetype); console.log('size: ' + (req.file.size / 1024).toFixed(2) + 'KB'); console.log('destination: ' + req.file.destination); console.log('filename: ' + req.file.filename); console.log('path: ' + req.file.path); });
由輸出可以看到,文件的命名使用一個哈希值表示,并且去除了后綴名,想要保持文件的原有的命名格式,需要再通過fs對文件進行改名。
app.post('/upload', upload.single('test-upload'), (req, res) => { // 沒有附帶文件 if (!req.file) { res.json({ ok: false }); return; } // 輸出文件信息 console.log('===================================================='); console.log('fieldname: ' + req.file.fieldname); console.log('originalname: ' + req.file.originalname); console.log('encoding: ' + req.file.encoding); console.log('mimetype: ' + req.file.mimetype); console.log('size: ' + (req.file.size / 1024).toFixed(2) + 'KB'); console.log('destination: ' + req.file.destination); console.log('filename: ' + req.file.filename); console.log('path: ' + req.file.path); // 重命名文件 let oldPath = path.join(__dirname, req.file.path); let newPath = path.join(__dirname, 'uploads/' + req.file.originalname); fs.rename(oldPath, newPath, (err) => { if (err) { res.json({ ok: false }); console.log(err); } else { res.json({ ok: true }); } }); });
完整代碼
ajax異步文件上傳、進度顯示、圖片預覽
<input type="file" name="file" id="file" /> <button id="upload">上傳</button> <span id="progress">0</span> <img id="image" src="" width="200" />
(function () { 'use strict'; var file = document.querySelector('#file'); var upload = document.querySelector('#upload'); var progress = document.querySelector('#progress'); var image = document.querySelector('#image'); var xhr = new XMLHttpRequest(); upload.addEventListener('click', uploadFile, false); file.addEventListener('change', previewImage, false); // 點擊上傳 function uploadFile(event) { var formData = new FormData(); formData.append('test-upload', file.files[0]); xhr.onload = uploadSuccess; xhr.upload.onprogress = setProgress; xhr.open('post', '/upload', true); xhr.send(formData); } // 成功上傳 function uploadSuccess(event) { if (xhr.readyState === 4) { console.log(xhr.responseText); } } // 進度條 function setProgress(event) { if (event.lengthComputable) { var complete = Number.parseInt(event.loaded / event.total * 100); progress.innerHTML = complete + '%'; } } // 圖片預覽 function previewImage(event) { var reader = new FileReader(); reader.onload = function (event) { image.src = event.target.result; }; reader.readAsDataURL(event.target.files[0]); } })();
express服務器提供文件上傳接口
const express = require('express'); const upload = require('multer')({ dest: 'uploads/' }); const path = require('path'); const fs = require('fs'); const port = 8080; let app = express(); app.set('port', port); // index.html, index.js放在static文件夾中 app.use(express.static(path.join(__dirname, 'static'))); app.get('*', (req, res) => { res.redirect('index.html'); }); app.post('/upload', upload.single('test-upload'), (req, res) => { // 沒有附帶文件 if (!req.file) { res.json({ ok: false }); return; } // 輸出文件信息 console.log('===================================================='); console.log('fieldname: ' + req.file.fieldname); console.log('originalname: ' + req.file.originalname); console.log('encoding: ' + req.file.encoding); console.log('mimetype: ' + req.file.mimetype); console.log('size: ' + (req.file.size / 1024).toFixed(2) + 'KB'); console.log('destination: ' + req.file.destination); console.log('filename: ' + req.file.filename); console.log('path: ' + req.file.path); // 重命名文件 let oldPath = path.join(__dirname, req.file.path); let newPath = path.join(__dirname, 'uploads/' + req.file.originalname); fs.rename(oldPath, newPath, (err) => { if (err) { res.json({ ok: false }); console.log(err); } else { res.json({ ok: true }); } }); }); app.listen(port, () => { console.log("[Server] localhost:" + port); });
以上就是本文的全部內(nèi)容,希望本文的內(nèi)容對大家的學習或者工作能帶來一定的幫助,同時也希望多多支持腳本之家!
- node+axios實現(xiàn)服務端文件上傳示例
- 利用node+koa+axios實現(xiàn)圖片上傳和回顯功能
- 詳解Vue+axios+Node+express實現(xiàn)文件上傳(用戶頭像上傳)
- nodejs+express實現(xiàn)文件上傳下載管理網(wǎng)站
- nodejs基于express實現(xiàn)文件上傳的方法
- 詳解nodejs實現(xiàn)本地上傳圖片并預覽功能(express4.0+)
- Nodejs+express+html5 實現(xiàn)拖拽上傳
- 基于nodejs+express(4.x+)實現(xiàn)文件上傳功能
- 使用express+multer實現(xiàn)node中的圖片上傳功能
- 使用nodejs+express實現(xiàn)簡單的文件上傳功能
- Nodejs進階:基于express+multer的文件上傳實例
- NodeJS實現(xiàn)圖片上傳代碼(Express)
- node+express+axios實現(xiàn)單文件上傳功能
相關文章
用js限制網(wǎng)頁只在微信瀏覽器中打開(或者只能手機端訪問)
這篇文章主要介紹了用js限制網(wǎng)頁只在微信瀏覽器中打開,很多電影站也是這么限制的,原因你懂的,需要的朋友可以參考下2020-01-01JS面向?qū)ο蠡A講解(工廠模式、構造函數(shù)模式、原型模式、混合模式、動態(tài)原型模式)
這篇文章主要介紹了面向?qū)ο驤S基礎講解,工廠模式、構造函數(shù)模式、原型模式、混合模式、動態(tài)原型模式,需要的朋友可以參考下2014-08-08將html頁面保存成圖片,圖片寫入pdf的實現(xiàn)方法(推薦)
下面小編就為大家?guī)硪黄獙tml頁面保存成圖片,圖片寫入pdf的實現(xiàn)方法(推薦)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-09-09ionic App問題總結系列之ionic點擊系統(tǒng)返回鍵退出App
本篇文章主要介紹了ionic App問題總結系列之ionic點擊系統(tǒng)返回鍵退出App,具有一定的參考價值,有興趣的可以了解一下2017-08-08