基于NodeJS+Express實(shí)現(xiàn)文件上傳下載

需求描述

  1. 允許用戶上傳文件,服務(wù)端從請求中讀取并且解析
  2. 允許用戶下載文件,如報表導(dǎo)出功能,基于當(dāng)前展示的表格數(shù)據(jù),導(dǎo)出excel文件

涉及到的技術(shù)點(diǎn)

  1. 文件讀寫
  2. 文件上傳
  3. 文件下載

文件上傳與讀取文件

前端代碼

比較常見的場景是:在前端頁面中會有一個表單,表單中會輸入一些字段,同時上傳文件。
代碼示例:

<form method="POST" action="/apps/rel" encType="multipart/form-data" name="form">
    <input id="inputFile" type="file"
           accept="application/vnd.ms-excel|xls"
           name="inputFile" onChange={this.handleFileChange}/>
    <button onClick={this.handleSubmit}>點(diǎn)擊提交</button>
</form>

表單提交一般通過js代碼提交:首先做數(shù)據(jù)合法性校驗,然后執(zhí)行form.submit()執(zhí)行提交。如果不想頁面跳轉(zhuǎn),則將表單target設(shè)置為一個隱藏的iframe。

服務(wù)端代碼

服務(wù)端處理上傳文件使用multer npm包。
代碼示例:

//先引入multer
var multer = require('multer');
var upload = multer();
//從upload中獲取文件

router.post('/rel', upload.fields([{name: 'inputFile', maxCount: 1}]),function(req,res,next){
     var inputFiles = req.files; //未傳時為undefined
     ...
     //讀取文件內(nèi)容
     var content = inputFiles['inputFile'][0].buffer.toString();

})

multer的更多使用請參考官方文檔。

文件下載

文件下載比較典型的例子是報表導(dǎo)出:通過數(shù)據(jù)庫數(shù)據(jù)查詢,寫入到csv文件中,然后支持瀏覽器下載。
代碼示例:

router.get('/export',(req,res,next)=>{
    //查詢數(shù)據(jù)庫
    ...
    let realPath = path.join(TMP_PATH, fileName);
    let writeStream = fs.createWriteStream(realPath);
    let bom = new Buffer('\xEF\xBB\xBF', 'binary');
    writeStream.write(bom);
    //寫入數(shù)據(jù)內(nèi)容
    ...

    writeStream.end();
    writeStream.on('finish', ()=> {
        res.download(realPath, outName, (err)=> {
            fs.unlink(realPath, (err2)=> {
                if (err2) {
                    logger.error('刪除文件失敗,err=' + err);
                }

            });
            if (err) {
                logger.error(err);
                res.status(404).end();
            }

        });
    });
})

常見問題

  1. 寫入csv中文亂碼問題
    由于 NodeJS只支持 ascii、utf8、base64、binary 編碼方式, 不支持 MS 的 utf-8 + BOM 格式, 所以 excel 打開中文亂碼。
    解決方法:既然excel需要BOM,寫入數(shù)據(jù)前先加入一個BOM。utf-8對應(yīng)的BOM是EF BB BF
 let writeStream = fs.createWriteStream(realPath);
 let bom = new Buffer('\xEF\xBB\xBF', 'binary');
 writeStream.write(bom);
 writeStream.write('hello');
 writeStream.write('繼續(xù)測試');
 writeStream.end('中文');
...
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,554評論 19 139
  • 本文包括:1、文件上傳概述2、利用 Commons-fileupload 組件實(shí)現(xiàn)文件上傳3、核心API——Dis...
    廖少少閱讀 12,744評論 5 91
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 179,034評論 25 709
  • 瑞恩放下手機(jī),低聲叫罵了一句:“該死,我怎么會有這么差的記性?”聲音雖然不大,但是仍然讓旁邊的吉姆聽到了。 “怎么...
    看笑話兒不嫌事大閱讀 563評論 0 0
  • 1 夏末的風(fēng)吹到你的琴聲 一個人在哭泣 在前往凄凄的秋夜里 2 我耿耿于懷,我這個夏季遇到的風(fēng)雨 不是純粹的天氣 ...
    徐夭夭閱讀 462評論 6 6

友情鏈接更多精彩內(nèi)容