最近在做一個文件下載的功能,這里把做的過程中用的技術(shù)和坑簡要總結(jié)下。
合浦網(wǎng)站制作公司哪家好,找成都創(chuàng)新互聯(lián)公司!從網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、APP開發(fā)、成都響應(yīng)式網(wǎng)站建設(shè)公司等網(wǎng)站項目制作,到程序開發(fā),運營維護。成都創(chuàng)新互聯(lián)公司于2013年成立到現(xiàn)在10年的時間,我們擁有了豐富的建站經(jīng)驗和運維經(jīng)驗,來保證我們的工作的順利進行。專注于網(wǎng)站建設(shè)就選成都創(chuàng)新互聯(lián)公司。
1. 單文件下載(a標(biāo)簽)
同源單文件
針對單文件的情況下,同源的文件,可以通過 < a> 標(biāo)簽的 download 屬性下載文件
const elt = document.createElement('a'); elt.setAttribute('href', url); elt.setAttribute('download', 'file.png'); elt.style.display = 'none'; document.body.appendChild(elt); elt.click(); document.body.removeChild(elt);
但是這個方案并不適用于非同源的資源,此時它相當(dāng)于普通的超鏈接,點擊會跳轉(zhuǎn)到資源頁面,而不是下載。
非同源圖片
如果不存在CORS問題, 可以借助Blob實現(xiàn)下載(構(gòu)造xhr請求文件地址, 以Blob的形式接收Response):
function downloadWithBlob(url) { fetch(url).then(res => res.blob().then(blob => { var a = document.createElement('a'); var url = window.URL.createObjectURL(blob); var filename = 'file.png'; a.href = url; a.download = filename; a.click(); window.URL.revokeObjectURL(url); })); }
如果存在CORS問題,可以考慮使用 canvas 將圖片轉(zhuǎn)換成 base64 編碼之后再通過 標(biāo)簽的 download 屬性下載
function downloadPic(url) { const img = new Image; const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); img.onload = function() { canvas.width = this.width; canvas.height = this.height; ctx.drawImage(this, 0, 0); const elt = document.createElement('a'); elt.setAttribute('href', canvas.toDataURL('image/png')); elt.setAttribute('download', 'file.png'); elt.style.display = 'none'; document.body.appendChild(elt); elt.click(); document.body.removeChild(elt); }; img.crossOrigin = 'anonymous'; img.src = url; }
2. 單文件下載(iframe)
iframe方式是在頁面內(nèi)隱藏iframe, 然后將下載地址加載到iframe中, 從而觸發(fā)瀏覽器的下載行為
const iframe = document.createElement('iframe'); iframe.src = url; iframe.style.display = 'none'; document.body.appendChild(iframe);
但是這里發(fā)現(xiàn),即使是同域的圖片,也無法完成下載,這是為啥呢?
這里就有個上面的a鏈接下載沒有提到的問題:什么樣的鏈接才能觸發(fā)瀏覽器的下載:
url如何觸發(fā)瀏覽器自動下載
一個url能否觸發(fā)瀏覽器自動下載,主要看該請求響應(yīng)頭response header是否滿足,一般是看Content-Disposition和Content-Type這兩個消息頭:
只要url滿足上述觸發(fā)的要求,那么都可以通過iframe的形式來下載
3. 代理服務(wù)處理下載
如果后端自己也能控制的話,或者后端能配合的話,可以寫一個代理服務(wù),在后端去請求文件數(shù)據(jù),然后設(shè)置好相應(yīng)的response header, 然后前端請求代理服務(wù)來做下載。
前端(假設(shè)代理服務(wù)接口是http://exampale.com/download):
const downloadUrl = 'http://exampale.com/download?url=' + encodeURIComponent(url) + '&name=xxx'; const elt = document.createElement('a'); elt.setAttribute('href', downloadUrl); elt.setAttribute('download', 'file.png'); ...
后端
const url = decodeURIComponent(req.query.url); http.get(url, (response) => { res.setHeader('Content-disposition', 'attachment;filename=' + req.query.name); res.setHeader('Content-type', 'application/octet-stream'); response.pipe(res); });
單文件的處理先寫到這里,多文件的下載下篇在寫。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。
文章標(biāo)題:通過JavaScript下載文件到本地的方法(單文件)
文章分享:http://m.2m8n56k.cn/article6/jogcig.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站設(shè)計公司、品牌網(wǎng)站建設(shè)、微信小程序、面包屑導(dǎo)航、標(biāo)簽優(yōu)化、ChatGPT
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:[email protected]。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)