中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

JS中以16進制字符串形式進行多文件上傳和下載的案例分析

發布時間:2020-12-03 13:23:44 來源:億速云 閱讀:426 作者:小新 欄目:web開發

這篇文章將為大家詳細講解有關JS中以16進制字符串形式進行多文件上傳和下載的案例分析,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

最近在維護一個比較老的 Web 項目,其中用到了 DWR 2.0 (一種可以在 js 里調用 Java 方法的遠程通信框架)。現在要利用這個框架實現上傳文件到服務端和從服務端下載文件,然而這個項目用的 DWR 2.0,默認只支持調用以基本數據類型,以及String、 List、Map 等常用類型作為參數和返回值的 Java 方法,無法使用 FileTransfer、InputStream、MultipartFile 這些對象。

既然可以傳遞字符串,那就采用文件與字符串互轉的方式進行前后交互。流程如下:
上傳文件時,文件 -> ArrayBuffer -> 16 進制字符串 -> byte[] -> 文件  
下載文件時,文件 -> byte[] -> 16 進制字符串 -> Uint8Array -> blob -> 文件

2. 上傳文件

HTML 代碼:

<input type="file" multiple="multiple" onchange="readFilesAndUpload(event)" />

JS 代碼:

// 將 ArrayBuffer 轉為 16 進制字符串
function bufToHex(buffer) {
    return Array.prototype.map.call(new Uint8Array(buffer), function (x) {
        return ('00' + x.toString(16)).slice(-2)
    }).join('')
}
function readFilesAndUpload(event) {
    var processed = 0
    var files = event.target.files
    var len = files.length
    var filenameArr = new Array(len)    // 文件名
    var fileContextArr = new Array(len)    // 文件內容
    for (var i = 0; i < len; ++i) {
        var reader = new FileReader()
        reader.index = i
        reader.filename = files[i].name
        reader.readAsArrayBuffer(files[i])    // 將文件讀到 ArrayBuffer
        reader.onload = function (e) {
            filenameArr[this.index] = this.filename
            fileContextArr[this.index] = bufToHex(this.result)

            // FileReader 以異步的方式讀取文件,需要借助外部變量判斷是否讀完全部文件
            if (++processed === len) {
                // 將 filenameArr 和 fileContext 上傳到服務端
            }
        }
    }
}

Java 代碼:

private static final String UPLOAD_DIR = "D://Files/";

public void uploadFiles(List<String> filenameArr, List<String> fileContextArr) throws IOException {
    byte[] bytes;
    FileOutputStream fos;
    for (int i = 0; i < filenameArr.size(); ++i) {
        String file = fileContextArr.get(i);
        
        // 將 16 進制字符串轉換成 byte[]
        bytes = new byte[file.length() / 2];
        for (int j = 0; j < file.length() / 2; ++j) {
            String subStr = file.substring(j * 2, j * 2 + 2);
            bytes[j] = (byte) Integer.parseInt(subStr, 16);
        }

        // 保存到本地磁盤
        fos = new FileOutputStream(UPLOAD_DIR + filenameArr.get(i), true);
        fos.write(bytes);
        fos.close();
    }
}

3. 下載文件

Java 代碼:

public String downloadFile(String filename) throws IOException {
    File file = new File(UPLOAD_DIR + filename);
    if (!file.exists()) {
        return null;
    }
    
    // 將文件讀到 byte[]
    byte[] buffer = new byte[(int) file.length()];
    InputStream is = new FileInputStream(file);
    is.read(buffer);
    is.close();

    // 將 byte[] 轉換成 16 進制字符串
    StringBuilder stringBuilder = new StringBuilder();
    for (int i = 0; i < buffer.length; i++) {
        int v = buffer[i] & 0xFF;
        String hv = Integer.toHexString(v);
        if (hv.length() < 2) {
            stringBuilder.append(0);
        }
        stringBuilder.append(hv);
    }
    return stringBuilder.toString();
}

JS 代碼:

// 16 進制字符串轉換成整型數組
function hexToBytes(hexStr) {
    var bytes = []
    for (var c = 0; c < hexStr.length; c += 2)
        bytes.push(parseInt(hexStr.substr(c, 2), 16))
    return bytes
}

function downloadFile() {
    // 調用服務端方法,取得 16 進制字符串 res
    var uint8Array = new Uint8Array(hexToBytes(res))
    var blob = new Blob([uint8Array], {type: "application/octet-stream"})

    // 兼容 IE、火狐和谷歌的下載方式
    if (window.navigator && window.navigator.msSaveOrOpenBlob) {
        window.navigator.msSaveOrOpenBlob(blob, filename)
    } else {
        var downloadElement = document.createElement("a")
        var href = window.URL.createObjectURL(blob)
        downloadElement.href = href
        downloadElement.download = filename
        document.body.appendChild(downloadElement)
        downloadElement.click()
        downloadElement.remove()
        window.URL.revokeObjectURL(href)
    }
}

關于JS中以16進制字符串形式進行多文件上傳和下載的案例分析就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

枣强县| 保德县| 龙口市| 漳州市| 突泉县| 治多县| 肥城市| 遵义县| 万全县| 江阴市| 汝州市| 波密县| 大姚县| 松桃| 高阳县| 荥阳市| 淮滨县| 抚州市| 永兴县| 盖州市| 曲阳县| 绵阳市| 永福县| 濉溪县| 汽车| 南雄市| 武安市| 通渭县| 淮安市| 彭水| 金溪县| 昌平区| 双峰县| 朝阳区| 道孚县| 赣榆县| 濮阳县| 临潭县| 鄂托克前旗| 郧西县| 五华县|