亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Android 文件分段上傳和下載實現(xiàn)方案

 更新時間:2024年11月22日 14:37:25   作者:粵M溫同學  
文章介紹了Android大文件下載和上傳的分段處理方法,在下載時,使用多線程并發(fā)請求文件的多個段,并將這些段寫入同一個文件,在上傳時,根據文件大小分段上傳,每段文件的大小由預設閾值決定,感興趣的朋友跟隨小編一起看看吧

一、背景

Android 中的大文件下載需要使用分段下載,下載通常是在線程中進行的,假如有5段,那同時5個線程去執(zhí)行下載,請求http返回文件流后,需要將多個文件流同時寫進同一個文件,這里用到

RandomAccessFile

分段上傳的話,只需要根據每段文件閥值,例如,50M為一段,將文件按照設置的閥值,分段上傳即可

二、相關代碼

2.1 分段上傳關鍵代碼

忽略網絡請求和狀態(tài)碼,每個人接口定義的的請求參數(shù)和返回code不一樣

       private val DEFAULT_BLOCK_SIZE: Long = 50 * 1024 * 1024 //50MB
        val blockSize=DEFAULT_BLOCK_SIZE
        val randomAccessFile = RandomAccessFile(filePath, "r")
        val fileLen = randomAccessFile.length()
        //超過設定的單個文件大小,需要分塊上傳
            val blockFileNum = Math.ceil((fileLen / blockSize.toDouble())).toInt()
            XLogUtil.d("${TAG}blockFileNum:$blockFileNum,,,,fileLen:$fileLen,,,blockSize:$blockSize,,,requestId:$requestId")
            var offSet = 0L
            var successNum = 0
            var isSendResult = true
            for (i in 0 until blockFileNum) {
                val startOffset = i * blockSize
                val blockFileLen = Math.min(blockSize, fileLen - startOffset)
                val fileData = getFileData(filePath, offSet, blockFileLen.toInt())
                // 創(chuàng)建文件名請求體
                val requestBody = RequestBody.create(null, fileData)
                val call = RetrofitClient.getUploadFileService(
                    token,
                    requestId,
                    offSet.toString(),
                    uploadType
                ).uploadFile(file.name, requestBody)
                XLogUtil.d("${TAG}upload 第${i + 1}塊 block file,offSet:$offSet,,,blockFileLen:$blockFileLen,,,blockFileNum:$blockFileNum,,,fileLen:$fileLen,,,filePath:$filePath,,,fileData size:${fileData?.size},,,requestId:$requestId")
                offSet += blockFileLen
                call.enqueue(object : Callback<ResponseBody?> {
                    override fun onResponse(
                        call: Call<ResponseBody?>,
                        response: Response<ResponseBody?>
                    ) {
                        val code = response.code()
                        XLogUtil.d("${TAG}upload 第${i + 1}塊 block file result code:$code,,,requestId:$requestId")
                        if (code == 201) {
                            //處理成功響應
                            successNum++
                            if (successNum == blockFileNum) {
                                XLogUtil.d("${TAG}upload all block file success,blockFileNum:$blockFileNum,,,requestId:$requestId")
                                listener?.apply {
                                    onSuccess(Constant.SUCCESS, requestId)
                                }
                                //上傳完
                            }
                        } else {
                            //處理失敗響應
                        }
                    }
                    override fun onFailure(call: Call<ResponseBody?>, t: Throwable) {
                        // 處理請求失敗
                        XLogUtil.d("${TAG}upload 第${i + 1}塊 block file onFailure message:${t.printStackTrace()}")
                    }
                })
            }
    /**
     * 根據偏移量獲取分塊文件數(shù)據
     */
    fun getFileData(filePath: String, offset: Long, length: Int): ByteArray? {
        // 使用RandomAccessFile隨機訪問文件
        var randomAccessFile: RandomAccessFile? = null
        try {
            randomAccessFile = RandomAccessFile(filePath, "r")
            val fileChannel = randomAccessFile.channel
            // 將文件的部分區(qū)域映射為內存區(qū)域
            val mappedByteBuffer =
                fileChannel.map(FileChannel.MapMode.READ_ONLY, offset, length.toLong())
            val data = ByteArray(length)
            // 從映射區(qū)域中讀取數(shù)據
            mappedByteBuffer[data]
            return data
        } catch (e: Exception) {
            e.printStackTrace()
            return null
        } finally {
            if (randomAccessFile != null) {
                try {
                    randomAccessFile.close()
                } catch (e: Exception) {
                    e.printStackTrace()
                }
            }
        }
    }

2.2、分段下載關鍵代碼

                val folder = File(DOWNLOAD_FOLDER_PATH)
                            if (!folder.exists()) {
                                folder.mkdirs()
                            }
                            val fileName = getFileNameWithPath(filePath)
                            val downFile = File(DOWNLOAD_FOLDER_PATH + File.separator + fileName)
                            if (!downFile.exists()) {
                                downFile.createNewFile()
                            }
                            // 使用輸入流保存響應體到文件,這里通常是通過http請求,返回的文件流,替換即可
                            val inputStream = body.byteStream()
                            val rw = RandomAccessFile(downFile, "rw")
                            rw.seek(startPosition)//文件寫入的初始位置
                            var hasReads = 0
                            var readLenght: Long = 0
                            val bytes = ByteArray(4096)
                            while ((inputStream.read(bytes).also { hasReads = it }) > 0) {
                                rw.write(bytes, 0, hasReads)
                                readLenght += hasReads
                                // val l = (readLenght * 100 / contentLength) as Int 單塊文件寫入進度
                            }
                            // 關閉文件輸出流和輸入流
                            inputStream.close()
                            rw.close()
    /**
     * 根據文件路徑獲取文件名
     */
    fun getFileNameWithPath(path: String): String {
        if (TextUtils.isEmpty(path)) {
            return ""
        }
        val start = path.lastIndexOf("/")
        return if (start != -1) {
            path.substring(start + 1)
        } else {
            "DEFAULT_NAME"
        }
    }

到此這篇關于Android 文件分段上傳和下載實現(xiàn)方案的文章就介紹到這了,更多相關Android 文件分段上傳和下載內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

最新評論