日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 >

Spring学习笔记(三十六)——SpringBoot 实现大文件分片上传、断点续传及秒传

發布時間:2024/8/1 41 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Spring学习笔记(三十六)——SpringBoot 实现大文件分片上传、断点续传及秒传 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

目錄

  • 文件分片上傳、斷點續傳及秒傳
    • 功能介紹
    • 相關插件技術介紹
    • 大文件上傳流程
  • 代碼實現
    • web端
      • 官方原生的案例修改
        • 效果如下
        • 代碼如下
      • 自定義uploader1
        • 效果如下
        • 代碼如下
      • 自定義uploader2
        • 效果如下
        • 代碼如下
    • SpringBoot實現后端
  • 源碼下載

文件分片上傳、斷點續傳及秒傳

功能介紹

  • 文件上傳
    小文件(圖片、文檔、視頻)上傳可以直接使用很多ui框架封裝的上傳組件,或者自己寫一個input 上傳,利用FormData 對象提交文件數據,后端使用spring提供的MultipartFile進行文件的接收,然后寫入即可。但是對于比較大的文件,比如上傳2G左右的文件(http上傳),就需要將文件分片上傳(file.slice()),否則中間http長時間連接可能會斷掉。

  • 分片上傳
    分片上傳,就是將所要上傳的文件,按照一定的大小,將整個文件分隔成多個數據塊(我們稱之為Part)來進行分別上傳,上傳完之后再由服務端對所有上傳的文件進行匯總整合成原始的文件。

  • 秒傳
    通俗的說,你把要上傳的東西上傳,服務器會先做MD5校驗,如果服務器上有一樣的東西,它就直接給你個新地址,其實你下載的都是服務器上的同一個文件,想要不秒傳,其實只要讓MD5改變,就是對文件本身做一下修改(改名字不行),例如一個文本文件,你多加幾個字,MD5就變了,就不會秒傳了.

  • 斷點續傳
    斷點續傳是在下載或上傳時,將下載或上傳任務(一個文件或一個壓縮包)人為的劃分為幾個部分,每一個部分采用一個線程進行上傳或下載,如果碰到網絡故障,可以從已經上傳或下載的部分開始繼續上傳或者下載未完成的部分,而沒有必要從頭開始上傳或者下載。本文的斷點續傳主要是針對斷點上傳場景。

  • 相關插件技術介紹

  • vue-simple-uploader
    前端使用vue-simple-uploader,一個基于simple-uploader封裝的上傳插件,imple-uploader.js(也稱 Uploader) 是一個上傳庫,支持多并發上傳,文件夾、拖拽、可暫停繼續、秒傳、分塊上傳、出錯自動重傳、手工重傳、進度、剩余時間、上傳速度等特性。
  • simple-uploader文檔案例:https://github.com/simple-uploader/vue-uploader
    vue-simple-uploader文檔案例:https://github.com/simple-uploader/Uploader/blob/develop/README_zh-CN.md

    使用前必須要了解的概念和方法
    相關概念
    chunkNumber: 當前塊的次序,第一個塊是 1,注意不是從 0 開始的。
    totalChunks: 文件被分成塊的總數。
    chunkSize: 分塊大小,根據 totalSize 和這個值你就可以計算出總共的塊數。注意最后一塊的大小可能會比這個要大。
    currentChunkSize: 當前塊的大小,實際大小。
    totalSize: 文件總大小。
    identifier: 這個就是MD5值,每個文件的唯一標示。
    filename: 文件名

    相關方法
    .upload() 開始或者繼續上傳。
    .pause() 暫停上傳。
    .resume() 繼續上傳。
    .cancel() 取消所有上傳文件,文件會被移除掉。
    .progress() 返回一個0-1的浮點數,當前上傳進度。
    .isUploading() 返回一個布爾值標示是否還有文件正在上傳中。
    .addFile(file) 添加一個原生的文件對象到上傳列表中。
    .removeFile(file) 從上傳列表中移除一個指定的 Uploader.File 實例對象。

  • MD5加密
    md5加密是可加鹽的非對稱加密算法。
    java使用MD5加密案例可以查看:https://qkongtao.cn/?p=580#h3-7
    web對文件的MD5加密可以使用:spark-md5
    spark-md5.js號稱是最適合前端最快的算法,能快速計算文件的md5。
  • 快速安裝:
    npm install --save spark-md5

    在組件中使用spark-md5時先引入:
    import SparkMD5 from 'spark-md5';

    spark-md5提供了兩個計算md5的方法。一種是用SparkMD5.hashBinary() 直接將整個文件的二進制碼傳入,直接返回文件的md5。這種方法對于小文件會比較有優勢——簡單而且速度超快。

    另一種方法是利用js中File對象的slice()方法(File.prototype.slice)將文件分片后逐個傳入spark.appendBinary()方法來計算、最后通過spark.end()方法輸出md5。很顯然,此方法就是我們前面講到的分片計算md5。這種方法對于大文件和超大文件會非常有利,不容易出錯,不占用大內存,并且能夠提供計算的進度信息。

    大文件上傳流程

  • 前端對文件進行MD5加密,并且將文件按一定的規則分片
  • vue-simple-uploader先會發送get請求校驗分片數據在服務端是否完整,如果完整則進行秒傳,如果不完整或者無數據,則進行分片上傳。
  • 后臺校驗MD5值,根據上傳的序號和分片大小計算相應的開始位置并寫入該分片數據到文件中。
  • 代碼實現

    web端

    源碼鏈接: https://gitee.com/KT1205529635/simple-uploader/tree/master/vue-uploader-master
    本次參考了官方文檔已經給位大佬的案例,根據自己的想法,實現了大文件的分片上傳、斷點續傳及秒傳
    其中前端寫了三個案例

    • 官方原生的案例修改
    • 自己根據插件提供的api和鉤子,自己diy自定義上傳(配合springboot后臺,文件夾上傳未作處理)
    • 自己diy自定義上傳的基礎上,在前端處理文件夾上傳(文件夾只接收文件夾里的所有文件,未處理文件夾相對目錄,可自己拓展)

    官方原生的案例修改

    效果如下

    代碼如下

    VueUploader.vue
    https://gitee.com/KT1205529635/simple-uploader/blob/master/vue-uploader-master/src/views/VueUploader.vue#

    <template><div class="container"><div class="logo"><img src="@/assets/logo.png" /></div><uploaderref="uploader":options="options":autoStart="false":file-status-text="fileStatusText"@file-added="onFileAdded"@file-success="onFileSuccess"@file-error="onFileError"@file-progress="onFileProgress"class="uploader-example"><uploader-unsupport></uploader-unsupport><uploader-drop><p>拖動文件到這里上傳</p><uploader-btn>選擇文件</uploader-btn><uploader-btn :directory="true">選擇文件夾</uploader-btn></uploader-drop><!-- uploader-list可自定義樣式 --><!-- <uploader-list></uploader-list> --><uploader-list><div class="file-panel" :class="{ collapse: collapse }"><div class="file-title"><p class="file-list-title">文件列表</p><div class="operate"><el-buttontype="text"@click="operate":title="collapse ? '折疊' : '展開'"><iclass="icon":class="collapse ? 'el-icon-caret-bottom' : 'el-icon-caret-top'"></i></el-button><el-button type="text" @click="close" title="關閉"><i class="icon el-icon-close"></i></el-button></div></div><ulclass="file-list":class="collapse ? 'uploader-list-ul-show' : 'uploader-list-ul-hidden'"><li v-for="file in uploadFileList" :key="file.id"><uploader-file:class="'file_' + file.id"ref="files":file="file":list="true"></uploader-file></li><div class="no-file" v-if="!uploadFileList.length"><i class="icon icon-empty-file"></i> 暫無待上傳文件</div></ul></div></uploader-list><span>下載</span></uploader></div> </template><script> import SparkMD5 from "spark-md5"; const FILE_UPLOAD_ID_KEY = "file_upload_id"; // 分片大小,20MB const CHUNK_SIZE = 20 * 1024 * 1024; export default {data() {return {options: {// 上傳地址target: "http://127.0.0.1:8025/api/upload",// 是否開啟服務器分片校驗。默認為 truetestChunks: true,// 真正上傳的時候使用的 HTTP 方法,默認 POSTuploadMethod: "post",// 分片大小chunkSize: CHUNK_SIZE,// 并發上傳數,默認為 3simultaneousUploads: 3,/*** 判斷分片是否上傳,秒傳和斷點續傳基于此方法* 這里根據實際業務來 用來判斷哪些片已經上傳過了 不用再重復上傳了 [這里可以用來寫斷點續傳!!!]*/checkChunkUploadedByResponse: (chunk, message) => {// message是后臺返回let messageObj = JSON.parse(message);let dataObj = messageObj.data;if (dataObj.uploaded !== undefined) {return dataObj.uploaded;}// 判斷文件或分片是否已上傳,已上傳返回 true// 這里的 uploadedChunks 是后臺返回]return (dataObj.uploadedChunks || []).indexOf(chunk.offset + 1) >= 0;},parseTimeRemaining: function (timeRemaining, parsedTimeRemaining) {//格式化時間return parsedTimeRemaining.replace(/\syears?/, "年").replace(/\days?/, "天").replace(/\shours?/, "小時").replace(/\sminutes?/, "分鐘").replace(/\sseconds?/, "秒");},},// 修改上傳狀態fileStatusTextObj: {success: "上傳成功",error: "上傳錯誤",uploading: "正在上傳",paused: "停止上傳",waiting: "等待中",},uploadIdInfo: null,uploadFileList: [],fileChunkList: [],collapse: true,};},created() {},methods: {onFileAdded(file, event) {console.log("file :>> ", file);// 有時 fileType為空,需截取字符console.log("文件類型:" + file.fileType);// 文件大小console.log("文件大小:" + file.size + "B");// 1. todo 判斷文件類型是否允許上傳// 2. 計算文件 MD5 并請求后臺判斷是否已上傳,是則取消上傳console.log("校驗MD5");this.getFileMD5(file, (md5) => {if (md5 != "") {// 修改文件唯一標識file.uniqueIdentifier = md5;// 請求后臺判斷是否上傳// 恢復上傳file.resume();}});},onFileSuccess(rootFile, file, response, chunk) {this.uploadFileList = this.$refs.uploader.fileList;console.log(this.uploadFileList);console.log("上傳成功");},onFileError(rootFile, file, message, chunk) {console.log("上傳出錯:" + message);},onFileProgress(rootFile, file, chunk) {console.log(`當前進度:${Math.ceil(file._prevProgress * 100)}%`);},// 計算文件的MD5值getFileMD5(file, callback) {let spark = new SparkMD5.ArrayBuffer();let fileReader = new FileReader();//獲取文件分片對象(注意它的兼容性,在不同瀏覽器的寫法不同)let blobSlice =File.prototype.slice ||File.prototype.mozSlice ||File.prototype.webkitSlice;// 當前分片下標let currentChunk = 0;// 分片總數(向下取整)let chunks = Math.ceil(file.size / CHUNK_SIZE);// MD5加密開始時間let startTime = new Date().getTime();// 暫停上傳file.pause();loadNext();// fileReader.readAsArrayBuffer操作會觸發onload事件fileReader.onload = function (e) {// console.log("currentChunk :>> ", currentChunk);spark.append(e.target.result);if (currentChunk < chunks) {currentChunk++;loadNext();} else {// 該文件的md5值let md5 = spark.end();console.log(`MD5計算完畢:${md5},耗時:${new Date().getTime() - startTime} ms.`);// 回調傳值md5callback(md5);}};fileReader.onerror = function () {this.$message.error("文件讀取錯誤");file.cancel();};// 加載下一個分片function loadNext() {const start = currentChunk * CHUNK_SIZE;const end =start + CHUNK_SIZE >= file.size ? file.size : start + CHUNK_SIZE;// 文件分片操作,讀取下一分片(fileReader.readAsArrayBuffer操作會觸發onload事件)fileReader.readAsArrayBuffer(blobSlice.call(file.file, start, end));}},fileStatusText(status, response) {if (status === "md5") {return "校驗MD5";} else {return this.fileStatusTextObj[status];}},/*** 折疊、展開面板動態切換*/operate() {if (this.collapse === false) {this.collapse = true;} else {this.collapse = false;}},/*** 關閉折疊面板*/close() {this.uploaderPanelShow = false;},}, }; </script><style lang="less" scoped> .logo {font-family: "Avenir", Helvetica, Arial, sans-serif;-webkit-font-smoothing: antialiased;-moz-osx-font-smoothing: grayscale;text-align: center;color: #2c3e50;margin-top: 60px; } .uploader-example {width: 880px;padding: 15px;margin: 40px auto 0;font-size: 12px;box-shadow: 0 0 10px rgba(0, 0, 0, 0.4); } .uploader-example .uploader-btn {margin-right: 4px; } .uploader-example .uploader-list {max-height: 440px;overflow: auto;overflow-x: hidden;overflow-y: auto; }#global-uploader {position: fixed;z-index: 20;right: 15px;bottom: 15px;width: 550px; }.file-panel {background-color: #fff;border: 1px solid #e2e2e2;border-radius: 7px 7px 0 0;box-shadow: 0 0 10px rgba(0, 0, 0, 0.2); }.file-title {display: flex;height: 60px;line-height: 30px;padding: 0 15px;border-bottom: 1px solid #ddd; }.file-title {background-color: #e7ecf2; }.uploader-file-meta {display: none !important; }.operate {flex: 1;text-align: right; }.file-list {position: relative;height: 240px;overflow-x: hidden;overflow-y: auto;background-color: #fff;padding: 0px;margin: 0 auto;transition: all 0.5s; }.uploader-file-size {width: 15% !important; }.uploader-file-status {width: 32.5% !important;text-align: center !important; }li {background-color: #fff;list-style-type: none; }.no-file {position: absolute;top: 50%;left: 50%;transform: translate(-50%, -50%);font-size: 16px; }/* 隱藏上傳按鈕 */ .global-uploader-btn {display: none !important;clip: rect(0, 0, 0, 0);/* width: 100px;height: 50px; */ }.file-list-title {/*line-height: 10px;*/font-size: 16px; }.uploader-file-name {width: 36% !important; }.uploader-file-actions {float: right !important; }.uploader-list-ul-hidden {height: 0px; } </style>

    自定義uploader1

    根據插槽和鉤子函數,實現自定義插件樣式,也實現簡單的下載。

    效果如下

    代碼如下

    DiyUpload1.vue
    https://gitee.com/KT1205529635/simple-uploader/blob/master/vue-uploader-master/src/views/DiyUpload1.vue#

    <template><div class="container"><div class="logo"><img src="@/assets/logo.png" /></div><uploaderref="uploader":options="options":autoStart="false":file-status-text="fileStatusText"@file-added="onFileAdded"@file-success="onFileSuccess"@file-error="onFileError"@file-progress="onFileProgress"class="uploader-example"><uploader-unsupport></uploader-unsupport><uploader-drop><p>拖動文件到這里上傳</p><uploader-btn>選擇文件</uploader-btn><!-- <uploader-btn :directory="true">選擇文件夾</uploader-btn> --></uploader-drop><!-- uploader-list可自定義樣式 --><!-- <uploader-list></uploader-list> --><uploader-list><div class="file-panel" :class="{ collapse: collapse }"><div class="file-title"><p class="file-list-title">文件列表</p><div class="operate"><el-buttontype="text"@click="operate":title="collapse ? '折疊' : '展開'"><iclass="icon":class="collapse ? 'el-icon-caret-bottom' : 'el-icon-caret-top'"></i></el-button><el-button type="text" @click="close" title="關閉"><i class="icon el-icon-close"></i></el-button></div></div><ulclass="file-list":class="collapse ? 'uploader-list-ul-show' : 'uploader-list-ul-hidden'"><li v-for="file in uploadFileList" :key="file.id"><!-- <uploader-file:class="'file_' + file.id"ref="files":file="file":list="true"></uploader-file> --><uploader-file :file="file" :list="true" ref="uploaderFile"><template slot-scope="props"><div class="filebox"><p class="fileNameBox"><span class="fileIcon"></span>{{ file.name }}</p><p class="fileProgressBox"><el-progressclass="progressLength":stroke-width="18":percentage="parseInt(props.progress.toFixed(2) * 100 - 1 < 0? 0: props.progress.toFixed(2) * 100)"></el-progress><spanclass="statusBtn progressBtn"v-if="!file.completed"@click="pause(file)"><iclass="el-icon-video-pause"v-if="!file.paused"title="暫停"></i><i class="el-icon-video-play" v-else title="繼續"></i></span><spanv-elseclass="downloadBtn progressBtn"@click="download(file)"><i class="el-icon-download" title="下載"></i></span><span class="cancelBtn progressBtn" @click="remove(file)"><i class="el-icon-error" title="刪除"></i></span></p><p class="fileInfoBox" v-if="!file.completed"><span class="fileInfoItem">速度:{{ props.formatedAverageSpeed }}</span><span class="fileInfoItem">已上傳:{{(parseFloat(props.formatedSize) * props.progress).toFixed(1)}}/{{ props.formatedSize }}</span><span class="fileInfoItem">剩余時間:{{ props.formatedTimeRemaining }}</span></p><p class="fileInfoBoxSuccess" v-else>上傳成功</p></div></template></uploader-file></li><div class="no-file" v-if="!uploadFileList.length"><i class="icon icon-empty-file"></i> 暫無待上傳文件</div></ul></div></uploader-list></uploader></div> </template><script> import SparkMD5 from "spark-md5"; const FILE_UPLOAD_ID_KEY = "file_upload_id"; // 分片大小,20MB const CHUNK_SIZE = 20 * 1024 * 1024; export default {data() {return {options: {// 上傳地址target: "http://127.0.0.1:8025/api/upload",// 是否開啟服務器分片校驗。默認為 truetestChunks: true,// 真正上傳的時候使用的 HTTP 方法,默認 POSTuploadMethod: "post",// 分片大小chunkSize: CHUNK_SIZE,// 并發上傳數,默認為 3simultaneousUploads: 3,/*** 判斷分片是否上傳,秒傳和斷點續傳基于此方法* 這里根據實際業務來 用來判斷哪些片已經上傳過了 不用再重復上傳了 [這里可以用來寫斷點續傳!!!]*/checkChunkUploadedByResponse: (chunk, message) => {// message是后臺返回let messageObj = JSON.parse(message);let dataObj = messageObj.data;if (dataObj.uploaded !== undefined) {return dataObj.uploaded;}// 判斷文件或分片是否已上傳,已上傳返回 true// 這里的 uploadedChunks 是后臺返回]return (dataObj.uploadedChunks || []).indexOf(chunk.offset + 1) >= 0;},parseTimeRemaining: function (timeRemaining, parsedTimeRemaining) {//格式化時間return parsedTimeRemaining.replace(/\syears?/, "年").replace(/\days?/, "天").replace(/\shours?/, "小時").replace(/\sminutes?/, "分鐘").replace(/\sseconds?/, "秒");},},// 修改上傳狀態fileStatusTextObj: {success: "上傳成功",error: "上傳錯誤",uploading: "正在上傳",paused: "停止上傳",waiting: "等待中",},uploadIdInfo: null,uploadFileList: [],fileChunkList: [],collapse: true,};},created() {},methods: {onFileAdded(file, event) {this.uploadFileList.push(file);console.log("file :>> ", file);// 有時 fileType為空,需截取字符console.log("文件類型:" + file.fileType);// 文件大小console.log("文件大小:" + file.size + "B");// 1. todo 判斷文件類型是否允許上傳// 2. 計算文件 MD5 并請求后臺判斷是否已上傳,是則取消上傳console.log("校驗MD5");this.getFileMD5(file, (md5) => {if (md5 != "") {// 修改文件唯一標識file.uniqueIdentifier = md5;// 請求后臺判斷是否上傳// 恢復上傳file.resume();}});},onFileSuccess(rootFile, file, response, chunk) {console.log("上傳成功");},onFileError(rootFile, file, message, chunk) {console.log("上傳出錯:" + message);},onFileProgress(rootFile, file, chunk) {console.log(`當前進度:${Math.ceil(file._prevProgress * 100)}%`);},// 計算文件的MD5值getFileMD5(file, callback) {let spark = new SparkMD5.ArrayBuffer();let fileReader = new FileReader();//獲取文件分片對象(注意它的兼容性,在不同瀏覽器的寫法不同)let blobSlice =File.prototype.slice ||File.prototype.mozSlice ||File.prototype.webkitSlice;// 當前分片下標let currentChunk = 0;// 分片總數(向下取整)let chunks = Math.ceil(file.size / CHUNK_SIZE);// MD5加密開始時間let startTime = new Date().getTime();// 暫停上傳file.pause();loadNext();// fileReader.readAsArrayBuffer操作會觸發onload事件fileReader.onload = function (e) {// console.log("currentChunk :>> ", currentChunk);spark.append(e.target.result);if (currentChunk < chunks) {currentChunk++;loadNext();} else {// 該文件的md5值let md5 = spark.end();console.log(`MD5計算完畢:${md5},耗時:${new Date().getTime() - startTime} ms.`);// 回調傳值md5callback(md5);}};fileReader.onerror = function () {this.$message.error("文件讀取錯誤");file.cancel();};// 加載下一個分片function loadNext() {const start = currentChunk * CHUNK_SIZE;const end =start + CHUNK_SIZE >= file.size ? file.size : start + CHUNK_SIZE;// 文件分片操作,讀取下一分片(fileReader.readAsArrayBuffer操作會觸發onload事件)fileReader.readAsArrayBuffer(blobSlice.call(file.file, start, end));}},fileStatusText(status, response) {if (status === "md5") {return "校驗MD5";} else {return this.fileStatusTextObj[status];}},/*** 折疊、展開面板動態切換*/operate() {if (this.collapse === false) {this.collapse = true;} else {this.collapse = false;}},/*** 關閉折疊面板*/close() {this.uploaderPanelShow = false;},// 點擊暫停pause(file, id) {console.log("file :>> ", file);if (file.paused) {file.resume();} else {file.pause();}},// 點擊刪除remove(file) {this.uploadFileList.findIndex((item, index) => {if (item.id === file.id) {this.$nextTick(() => {this.uploadFileList.splice(index, 1);});return;}});},// 點擊下載download(file, id) {console.log("file:>> ", file);window.location.href = `http://127.0.0.1:8025/api/download/${file.uniqueIdentifier}/${file.name}`;},}, }; </script><style lang="less" scoped> .logo {font-family: "Avenir", Helvetica, Arial, sans-serif;-webkit-font-smoothing: antialiased;-moz-osx-font-smoothing: grayscale;text-align: center;color: #2c3e50;margin-top: 60px; } .uploader-example {width: 880px;padding: 15px;margin: 40px auto 0;font-size: 12px;box-shadow: 0 0 10px rgba(0, 0, 0, 0.4); } .uploader-example .uploader-btn {margin-right: 4px; } .uploader-example .uploader-list {max-height: 440px;overflow: auto;overflow-x: hidden;overflow-y: auto; }#global-uploader {position: fixed;z-index: 20;right: 15px;bottom: 15px;width: 550px; }.file-panel {background-color: #fff;border: 1px solid #e2e2e2;border-radius: 7px 7px 0 0;box-shadow: 0 0 10px rgba(0, 0, 0, 0.2); }.file-title {display: flex;height: 60px;line-height: 30px;padding: 0 15px;border-bottom: 1px solid #ddd; }.file-title {background-color: #e7ecf2; } .uploader-file {height: 90px; }.uploader-file-meta {display: none !important; }.operate {flex: 1;text-align: right; }.file-list {position: relative;height: 300px;overflow-x: hidden;overflow-y: auto;background-color: #fff;padding: 0px;margin: 0 auto;transition: all 0.5s; }.uploader-file-size {width: 15% !important; }.uploader-file-status {width: 32.5% !important;text-align: center !important; }li {background-color: #fff;list-style-type: none; }.no-file {position: absolute;top: 50%;left: 50%;transform: translate(-50%, -50%);font-size: 16px; }.file-list-title {/*line-height: 10px;*/font-size: 16px; }.uploader-file-name {width: 36% !important; }.uploader-file-actions {float: right !important; }.uploader-list-ul-hidden {height: 0px; }.filebox {width: 100%;height: 60px; } .fileNameBox {width: 85%;margin: 0;padding: 0;font-size: 16px;margin-top: 5px;height: 30px;line-height: 30px;text-align: center; } .fileProgressBox {margin: 0;padding: 0;height: 20px;line-height: 20px;margin-top: 5px;margin-left: 10px;width: 100%; } /deep/ .el-progress-bar {width: 95%; } .progressLength {display: inline-block;line-height: 20px;width: 80%; } .progressBtn {margin-top: -5px;position: absolute;display: inline-block;font-size: 36px;margin-left: 10px;cursor: pointer; } .statusBtn {right: 90px;color: #ffba00; } .statusBtn:hover {color: #ffc833; } .cancelBtn {right: 30px;color: #ff4949; } .cancelBtn {margin-left: 10px; } .cancelBtn:hover {color: #ff6d6d; } .downloadBtn {right: 90px;color: #67c23a; } .downloadBtn:hover {color: #85ce61; } .fileInfoBox {margin: 0;padding: 0;font-size: 16px;width: 100%;height: 30px;line-height: 30px;margin-left: 10px;margin-bottom: 5px;.fileInfoItem {display: inline-block;width: 33%;} } .fileInfoBoxSuccess {margin: 0;padding: 0;font-size: 16px;width: 85%;height: 30px;line-height: 30px;margin-bottom: 5px;text-align: center; } </style>

    自定義uploader2

    在自定義uploader1上實現可上傳文件夾

    效果如下

    代碼如下

    https://gitee.com/KT1205529635/simple-uploader/blob/master/vue-uploader-master/src/views/DiyUpload2.vue#

    SpringBoot實現后端

    源碼鏈接: https://gitee.com/KT1205529635/simple-uploader/tree/master/springboot-upload-master
    后端實現簡單粗暴:springboot + jpa + hutool + mysql
    主要實現:

  • get請求接口校驗上傳文件MD5值和文件是否完整
  • post請求接收上傳文件,并且計算分片,寫入合成文件
  • 文件完整上傳完成時,往文件存儲表tool_local_storage中加一條該文件的信息
  • get請求接口實現簡單的文件下載
  • 目錄結構如下:

    關鍵代碼如下:

  • sql如下
  • DROP TABLE IF EXISTS `file_chunk`; CREATE TABLE `file_chunk` (`id` bigint(20) UNSIGNED NOT NULL AUTO_INCREMENT,`file_name` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '文件名',`chunk_number` int(11) NULL DEFAULT NULL COMMENT '當前分片,從1開始',`chunk_size` float NULL DEFAULT NULL COMMENT '分片大小',`current_chunk_size` float NULL DEFAULT NULL COMMENT '當前分片大小',`total_size` double(20, 0) NULL DEFAULT NULL COMMENT '文件總大小',`total_chunk` int(11) NULL DEFAULT NULL COMMENT '總分片數',`identifier` varchar(45) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '文件標識',`relative_path` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT 'md5校驗碼',`createtime` timestamp(0) NOT NULL DEFAULT CURRENT_TIMESTAMP,`updatetime` timestamp(0) NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP(0),PRIMARY KEY (`id`) USING BTREE ) ENGINE = InnoDB AUTO_INCREMENT = 1529 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Dynamic;-- ---------------------------- -- Table structure for tool_local_storage -- ---------------------------- DROP TABLE IF EXISTS `tool_local_storage`; CREATE TABLE `tool_local_storage` (`id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',`real_name` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '文件真實的名稱',`name` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '文件名',`suffix` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '后綴',`path` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '路徑',`type` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '類型',`size` varchar(100) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '大小',`identifier` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT 'md5校驗碼\r\n',`create_by` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '創建者',`update_by` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '更新者',`createtime` timestamp(0) NOT NULL DEFAULT CURRENT_TIMESTAMP,`updatetime` timestamp(0) NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP(0),PRIMARY KEY (`id`) USING BTREE ) ENGINE = InnoDB AUTO_INCREMENT = 3360 CHARACTER SET = utf8 COLLATE = utf8_general_ci COMMENT = '文件存儲' ROW_FORMAT = Compact;
  • controller實現
  • package cn.kt.springbootuploadmaster.controller;import cn.kt.springbootuploadmaster.domin.FileChunkParam; import cn.kt.springbootuploadmaster.domin.ResultVO; import cn.kt.springbootuploadmaster.service.FileChunkService; import cn.kt.springbootuploadmaster.service.FileService; import cn.kt.springbootuploadmaster.service.LocalStorageService; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.*;import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; import java.util.HashMap; import java.util.List; import java.util.Map;/*** Created by tao.* Date: 2022/6/29 11:56* 描述:*/ @RestController @Slf4j @RequestMapping("/api") public class FileUploadController {@Autowiredprivate FileService fileService;@Autowiredprivate FileChunkService fileChunkService;@Autowiredprivate LocalStorageService localStorageService;@GetMapping("/upload")public ResultVO<Map<String, Object>> checkUpload(FileChunkParam param) {log.info("文件MD5:" + param.getIdentifier());List<FileChunkParam> list = fileChunkService.findByMd5(param.getIdentifier());Map<String, Object> data = new HashMap<>(1);// 判斷文件存不存在if (list.size() == 0) {data.put("uploaded", false);return new ResultVO<>(200, "上傳成功", data);}// 處理單文件if (list.get(0).getTotalChunks() == 1) {data.put("uploaded", true);data.put("url", "");return new ResultVO<Map<String, Object>>(200, "上傳成功", data);}// 處理分片int[] uploadedFiles = new int[list.size()];int index = 0;for (FileChunkParam fileChunkItem : list) {uploadedFiles[index] = fileChunkItem.getChunkNumber();index++;}data.put("uploadedChunks", uploadedFiles);return new ResultVO<Map<String, Object>>(200, "上傳成功", data);}@PostMapping("/upload")public ResultVO chunkUpload(FileChunkParam param) {log.info("上傳文件:{}", param);boolean flag = fileService.uploadFile(param);if (!flag) {return new ResultVO(211, "上傳失敗");}return new ResultVO(200, "上傳成功");}@GetMapping(value = "/download/{md5}/{name}")public void downloadbyname(HttpServletRequest request, HttpServletResponse response, @PathVariable String name, @PathVariable String md5) throws IOException {localStorageService.downloadByName(name, md5, request, response);}}
  • FileService實現
    FileService.java
  • package cn.kt.springbootuploadmaster.service;import cn.kt.springbootuploadmaster.domin.FileChunkParam;/*** Created by tao.* Date: 2022/6/29 11:22* 描述:*/ public interface FileService {/*** 上傳文件* @param param 參數* @return*/boolean uploadFile(FileChunkParam param); }

    FileServiceImpl.java

    package cn.kt.springbootuploadmaster.service.impl;import cn.kt.springbootuploadmaster.domin.FileChunkParam; import cn.kt.springbootuploadmaster.enums.MessageEnum; import cn.kt.springbootuploadmaster.exception.BusinessException; import cn.kt.springbootuploadmaster.repository.LocalStorageRepository; import cn.kt.springbootuploadmaster.service.FileChunkService; import cn.kt.springbootuploadmaster.service.FileService; import cn.kt.springbootuploadmaster.service.LocalStorageService; import cn.kt.springbootuploadmaster.utils.FileUtil; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Service; import sun.misc.Cleaner;import java.io.File; import java.io.IOException; import java.io.RandomAccessFile; import java.lang.reflect.Method; import java.nio.MappedByteBuffer; import java.nio.channels.FileChannel; import java.security.AccessController; import java.security.PrivilegedAction;/*** Created by tao.* Date: 2022/6/29 11:22* 描述:*/ @Service("fileService") @Slf4j public class FileServiceImpl implements FileService {/*** 默認的分片大小:20MB*/public static final long DEFAULT_CHUNK_SIZE = 20 * 1024 * 1024;@Value("${file.BASE_FILE_SAVE_PATH}")private String BASE_FILE_SAVE_PATH;@Autowiredprivate FileChunkService fileChunkService;@Autowiredprivate LocalStorageService localStorageService;@Overridepublic boolean uploadFile(FileChunkParam param) {if (null == param.getFile()) {throw new BusinessException(MessageEnum.UPLOAD_FILE_NOT_NULL);}// 判斷目錄是否存在,不存在則創建目錄File savePath = new File(BASE_FILE_SAVE_PATH);if (!savePath.exists()) {boolean flag = savePath.mkdirs();if (!flag) {log.error("保存目錄創建失敗");return false;}}// todo 處理文件夾上傳(上傳目錄下新建上傳的文件夾)/*String relativePath = param.getRelativePath();if (relativePath.contains("/") || relativePath.contains(File.separator)) {String div = relativePath.contains(File.separator) ? File.separator : "/";String tempPath = relativePath.substring(0, relativePath.lastIndexOf(div));savePath = new File(BASE_FILE_SAVE_PATH + File.separator + tempPath);if (!savePath.exists()) {boolean flag = savePath.mkdirs();if (!flag) {log.error("保存目錄創建失敗");return false;}}}*/// 這里可以使用 uuid 來指定文件名,上傳完成后再重命名,File.separator指文件目錄分割符,win上的"\",Linux上的"/"。String fullFileName = savePath + File.separator + param.getFilename();// 單文件上傳if (param.getTotalChunks() == 1) {return uploadSingleFile(fullFileName, param);}// 分片上傳,這里使用 uploadFileByRandomAccessFile 方法,也可以使用 uploadFileByMappedByteBuffer 方法上傳boolean flag = uploadFileByRandomAccessFile(fullFileName, param);if (!flag) {return false;}// 保存分片上傳信息fileChunkService.saveFileChunk(param);return true;}private boolean uploadFileByRandomAccessFile(String resultFileName, FileChunkParam param) {try (RandomAccessFile randomAccessFile = new RandomAccessFile(resultFileName, "rw")) {// 分片大小必須和前端匹配,否則上傳會導致文件損壞long chunkSize = param.getChunkSize() == 0L ? DEFAULT_CHUNK_SIZE : param.getChunkSize().longValue();// 偏移量long offset = chunkSize * (param.getChunkNumber() - 1);// 定位到該分片的偏移量randomAccessFile.seek(offset);// 寫入randomAccessFile.write(param.getFile().getBytes());} catch (IOException e) {log.error("文件上傳失敗:" + e);return false;}return true;}private boolean uploadFileByMappedByteBuffer(String resultFileName, FileChunkParam param) {// 分片上傳try (RandomAccessFile randomAccessFile = new RandomAccessFile(resultFileName, "rw");FileChannel fileChannel = randomAccessFile.getChannel()) {// 分片大小必須和前端匹配,否則上傳會導致文件損壞long chunkSize = param.getChunkSize() == 0L ? DEFAULT_CHUNK_SIZE : param.getChunkSize().longValue();// 寫入文件long offset = chunkSize * (param.getChunkNumber() - 1);byte[] fileBytes = param.getFile().getBytes();MappedByteBuffer mappedByteBuffer = fileChannel.map(FileChannel.MapMode.READ_WRITE, offset, fileBytes.length);mappedByteBuffer.put(fileBytes);// 釋放unmap(mappedByteBuffer);} catch (IOException e) {log.error("文件上傳失敗:" + e);return false;}return true;}private boolean uploadSingleFile(String resultFileName, FileChunkParam param) {File saveFile = new File(resultFileName);try {// 寫入param.getFile().transferTo(saveFile);localStorageService.saveLocalStorage(param);} catch (IOException e) {log.error("文件上傳失敗:" + e);return false;}return true;}/*** 釋放 MappedByteBuffer* 在 MappedByteBuffer 釋放后再對它進行讀操作的話就會引發 jvm crash,在并發情況下很容易發生* 正在釋放時另一個線程正開始讀取,于是 crash 就發生了。所以為了系統穩定性釋放前一般需要檢* 查是否還有線程在讀或寫* 來源:https://my.oschina.net/feichexia/blog/212318** @param mappedByteBuffer mappedByteBuffer*/public static void unmap(final MappedByteBuffer mappedByteBuffer) {try {if (mappedByteBuffer == null) {return;}mappedByteBuffer.force();AccessController.doPrivileged((PrivilegedAction<Object>) () -> {try {Method getCleanerMethod = mappedByteBuffer.getClass().getMethod("cleaner");getCleanerMethod.setAccessible(true);Cleaner cleaner =(Cleaner) getCleanerMethod.invoke(mappedByteBuffer, new Object[0]);cleaner.clean();} catch (Exception e) {log.error("MappedByteBuffer 釋放失敗:" + e);}System.out.println("clean MappedByteBuffer completed");return null;});} catch (Exception e) {log.error("unmap error:" + e);}} }

    其他實現的細節可自己查看源碼,也可以根據自己的想法在這個demo中進行拓展。理清楚其中的大文件傳輸、秒傳、斷點續傳后,自己開發一個小網盤也不是什么難事了 _

    源碼下載

    https://gitee.com/KT1205529635/simple-uploader

    總結

    以上是生活随笔為你收集整理的Spring学习笔记(三十六)——SpringBoot 实现大文件分片上传、断点续传及秒传的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    操操操日日 | 久久狠狠亚洲综合 | 亚洲精品网站 | 午夜国产福利在线 | 69精品 | 午夜精品视频一区二区三区在线看 | 国产日韩视频在线播放 | 日韩大片在线免费观看 | 在线成人免费电影 | 欧美亚洲免费在线一区 | 97国产视频| 久久综合加勒比 | 国产亚洲精品成人 | 亚洲最新av在线网站 | 五月天天在线 | 91av短视频| 亚洲午夜av电影 | 成人禁用看黄a在线 | 超碰在线观看99 | 人人爽人人爽人人爽人人爽 | 99久久精品国产观看 | 欧美激精品 | 亚洲精品午夜一区人人爽 | 久草在线高清 | 91精品久久香蕉国产线看观看 | 96视频免费在线观看 | 欧美 日韩 国产 中文字幕 | 超级碰碰碰视频 | 日本在线观看一区 | 久久免费精品视频 | 一区二区三区高清不卡 | 久久丁香网 | 亚州国产精品久久久 | 欧亚日韩精品一区二区在线 | 欧美精品久久久久久久亚洲调教 | 日韩在线不卡视频 | 久久欧美在线电影 | 五月婷婷六月丁香在线观看 | 免费看高清毛片 | 国产精品久久9 | 亚洲国产精品影院 | 一区二区精品在线视频 | 亚洲精品网站 | 国产第页 | 日本精品二区 | 中国美女一级看片 | 国产又粗又猛又黄 | 激情网站免费观看 | 亚州av网站 | 91av网站在线观看 | 黄色精品国产 | 亚洲激情 | 四虎影视精品成人 | 亚州av网站| 手机在线小视频 | 国产精品欧美久久久久天天影视 | 99色亚洲| 精品视频123区在线观看 | 麻豆精品在线视频 | 国产精品嫩草影院123 | 免费观看丰满少妇做爰 | 国产精品18毛片一区二区 | 黄在线免费看 | 美女视频黄是免费的 | 日韩在线免费视频观看 | 黄色在线网站噜噜噜 | 国产精品电影一区二区 | 91精品国产99久久久久久红楼 | 91色偷偷 | 久久好看免费视频 | 欧美日韩观看 | 国产青春久久久国产毛片 | 波多野结衣电影久久 | 国产日韩欧美精品在线观看 | 久久福利 | 日韩免费大片 | av不卡在线看 | 五月婷婷开心中文字幕 | 97在线免费视频 | 日本成人中文字幕在线观看 | 国产又粗又猛又色又黄网站 | 国产精品手机在线播放 | 精品国产一区二区三区久久久蜜臀 | 久久天天躁夜夜躁狠狠85麻豆 | 日色在线视频 | 久久久久久久久久久久99 | 91福利视频免费观看 | 不卡精品 | 97夜夜澡人人双人人人喊 | 久久激情日本aⅴ | 在线国产精品一区 | 日日夜夜天天久久 | 日本视频高清 | 国产亚洲亚洲 | 久久视精品 | 成人97视频一区二区 | 日韩电影中文字幕 | 97精品国产aⅴ | av在线网站观看 | 国产婷婷在线观看 | 在线观看国产v片 | 日本精品久久久一区二区三区 | 五月天六月色 | 婷婷久久亚洲 | 狠狠躁日日躁狂躁夜夜躁av | 国产成人精品国内自产拍免费看 | 亚洲免费成人av电影 | 特黄特色特刺激视频免费播放 | 久久99日韩| 国产精品资源网 | 免费av高清| 国产伦精品一区二区三区… | 伊人婷婷网 | 日韩激情片在线观看 | 国产免费久久av | 在线播放 一区 | 欧美一级日韩三级 | 成人免费共享视频 | 色永久免费视频 | 四虎视频| 激情综合网天天干 | 久久精品国产一区二区三区 | 丁香高清视频在线看看 | 精品国产一区二区三区在线观看 | 综合久久网 | 成人免费视频在线观看 | 毛片一区二区 | 久久视频国产精品免费视频在线 | 久久这里有精品 | 国产精品国产三级国产aⅴ无密码 | 天天摸天天操天天舔 | www.伊人网 | 天天操天天爽天天干 | 日韩在线观看中文 | 亚洲综合视频网 | 在线看国产视频 | 久久久久久久久久久久av | 成人国产精品免费 | 亚洲成人免费在线 | 国产福利在线 | 精品一二三四五区 | 国产亚洲免费观看 | 五月天天色| 久精品视频在线观看 | 免费福利视频网站 | 久久经典国产视频 | 亚洲成人精品在线 | 日夜夜精品视频 | 国产成人在线观看 | 亚洲精品456在线播放 | 久久噜噜少妇网站 | 狠狠网站| 欧美精品一区二区在线观看 | 免费网站黄 | 国产黄色资源 | 日韩精品中文字幕在线播放 | 在线天堂中文在线资源网 | 精品国产123 | 最近中文字幕完整视频高清1 | 三级a毛片 | 在线观看中文字幕 | 国产色在线 | 国产精品福利在线播放 | 免费a现在观看 | 亚洲成人av影片 | 精品国产视频在线观看 | 啪啪激情网 | 夜夜骑首页 | 丁香五香天综合情 | 亚洲天堂精品视频在线观看 | 日韩成人免费在线观看 | 欧美一级日韩免费不卡 | 亚洲精品视频网站在线观看 | 国产精品成人一区二区 | 午夜精品成人一区二区三区 | 成人午夜电影在线 | 国产精品白虎 | 久久久久亚洲天堂 | 在线观看的a站 | av在线网站大全 | 三级a视频 | 狠狠久久 | 91九色综合| 天天操天天色天天射 | 青青河边草免费观看 | 在线观看视频一区二区三区 | 久久综合影音 | 香蕉97视频观看在线观看 | 怡红院av久久久久久久 | 国产在线精品一区二区不卡了 | 亚洲少妇xxxx | 亚洲精品免费播放 | 中文视频在线看 | 国产精品免费不卡 | 黄色软件在线观看 | 国产精品久久久久久超碰 | 激情综合色综合久久综合 | 91精品在线视频观看 | 三级性生活视频 | 成人国产精品免费 | 国产99久久久国产 | 欧美成人tv | 97超级碰| 国产精品毛片完整版 | 首页中文字幕 | 狠狠狠色 | 日韩av一区二区三区在线观看 | 五月天婷婷综合 | 成片免费 | 亚洲精品视频在线观看视频 | 久草视频看看 | 一二三久久久 | 四虎在线永久免费观看 | 丁香视频在线观看 | 成人app在线免费观看 | 久草男人天堂 | 狠狠狠狠狠狠狠狠干 | 免费在线黄色av | 亚洲在线精品 | 国产精品av免费在线观看 | 国产剧情一区二区在线观看 | 91麻豆精品国产91久久久更新时间 | 精品国偷自产国产一区 | 91麻豆精品国产自产在线游戏 | 激情五月激情综合网 | 天天舔天天搞 | 国产精品视频线看 | 人人澡超碰碰97碰碰碰软件 | 91久久电影 | 亚洲.www | 欧美日在线观看 | 日韩精品91偷拍在线观看 | 中文字幕一区二区三区久久 | 97超碰超碰久久福利超碰 | 国产精品乱码在线 | 久久人人爽爽人人爽人人片av | 天天射天天射 | 成人黄色视 | 久久爱资源网 | 日韩精品中文字幕有码 | 免费在线电影网址大全 | 日韩精品一区不卡 | 色五月成人 | 天天做天天干 | 精品久久91 | 欧美日韩视频一区二区 | 久久久网站 | 色国产精品一区在线观看 | 日韩特黄一级欧美毛片特黄 | 精品久久久久久久久久岛国gif | 97色噜噜| 在线观看视频h | 欧美日韩免费一区二区 | 一区二区av| 国产九色视频在线观看 | 欧美热久久 | 欧洲视频一区 | 美女网站视频免费都是黄 | 能在线看的av | 日本中文在线观看 | 热re99久久精品国产66热 | 最近中文字幕高清字幕免费mv | 在线观看va | 91夫妻自拍| 亚洲精品五月 | 五月激情久久久 | 久久精品视频在线看 | 天天插狠狠插 | 婷婷在线视频观看 | 国产v欧美 | 久久国产精品免费看 | 综合伊人久久 | 中文字幕三区 | 免费欧美精品 | 特黄一级毛片 | 精品自拍sae8—视频 | 日本三级在线观看中文字 | 色99色 | 天堂v中文 | 综合色伊人 | av成人资源 | 欧美美女视频在线观看 | 亚洲国产精品日韩 | 日本不卡123 | 99久久精品视频免费 | 精品一区二区综合 | 黄色在线观看免费 | 久久国内精品99久久6app | 在线观看香蕉视频 | 国产色网站 | 热久久视久久精品18亚洲精品 | 久久久久久久电影 | 日韩精品一区二区三区中文字幕 | 探花视频在线观看免费 | 99看视频在线观看 | 热久在线 | 成人啪啪18免费游戏链接 | 国产成人免费网站 | 黄色大片视频网站 | 天天色成人 | 91av在线免费看 | 国产一区二区日本 | 久久久亚洲成人 | 天天拍天天爽 | 亚洲欧洲精品一区二区精品久久久 | 不卡av电影在线 | 久久99精品久久久久婷婷 | 中文字幕在线日本 | 射综合网 | 国产亚洲综合在线 | 丁香资源影视免费观看 | av在线免费在线观看 | 欧美日韩视频网站 | 九九综合在线 | 亚洲日韩精品欧美一区二区 | 国产又黄又爽无遮挡 | 国产一区二区高清不卡 | 欧美伦理一区二区 | 国产手机免费视频 | 精品一区二区影视 | 精品国产乱码久久久久久1区二区 | h网站免费在线观看 | 国产精品一区二区三区久久久 | 中文字幕在线日亚洲9 | 精品国产一区二 | 国产清纯在线 | 国产在线观看国语版免费 | 久久亚洲精品电影 | 欧美淫视频 | 91在线亚洲 | 亚洲高清在线 | 日韩精品久久久久久久电影竹菊 | 成年人免费看片 | 婷婷在线免费观看 | 亚洲人成在线电影 | 国产一在线精品一区在线观看 | 成人黄色一级视频 | 六月丁香婷婷在线 | 激情欧美日韩一区二区 | 亚洲国产精品va在线看黑人 | 狠狠狠色丁香综合久久天下网 | 精品福利在线视频 | 欧美成人在线网站 | 亚洲国产资源 | 日韩欧美一区二区三区黑寡妇 | 成人国产精品一区二区 | 91精品在线观看入口 | 日韩在线观看小视频 | 精品二区久久 | 久久精品精品电影网 | 欧美精品免费在线观看 | 四虎影院在线观看av | 久久久麻豆精品一区二区 | 亚洲高清在线观看视频 | 精品1区2区 | 欧洲色综合 | 色网站在线观看 | 正在播放五月婷婷狠狠干 | 国产最顶级的黄色片在线免费观看 | 丁香av| 超级碰碰免费视频 | 久草视频网| 亚洲三级黄色 | 日韩av网址在线 | 四虎在线影视 | av在线小说 | 夜夜骑天天操 | 91在线产啪 | 国产精品麻豆三级一区视频 | 中文字幕精品三区 | 在线视频亚洲 | 久久国产精品99久久久久久丝袜 | 亚洲综合欧美激情 | 丁香高清视频在线看看 | 久久久久久国产精品免费 | 伊人伊成久久人综合网站 | 啪嗒啪嗒免费观看完整版 | 六月丁香伊人 | 成年人免费av | 伊人狠狠色丁香婷婷综合 | 99久久综合国产精品二区 | 国产亚洲精品久久久久久移动网络 | 天天色天天干天天色 | 亚洲成人资源在线观看 | 一级国产视频 | 国产精品国产亚洲精品看不卡15 | 黄色综合 | a级国产乱理论片在线观看 特级毛片在线观看 | 丁香九月激情综合 | 国产福利久久 | 中文字幕资源网 国产 | 亚洲免费在线 | 日本中文字幕免费观看 | 激情九九| 日韩高清在线看 | 成人一区电影 | 久久天堂亚洲 | 午夜av在线免费 | 日本中文字幕一二区观 | 69国产精品视频 | 欧美亚洲一级片 | 亚洲免费小视频 | av中文在线播放 | 超黄视频网站 | 久久图| 久草在线手机视频 | 欧美日韩国产在线精品 | 天天看天天干 | 中文字幕在线观看完整版电影 | 免费观看第二部31集 | 啪啪免费观看网站 | 国产高清在线免费视频 | 亚洲视频,欧洲视频 | 免费在线观看污网站 | 91 在线视频播放 | 欧美一区二区三区在线视频观看 | 久久久久久久99 | 色综合久久中文综合久久牛 | 国产91小视频 | 久久综合射| 亚洲天堂网在线视频 | 国产精品黄色影片导航在线观看 | 色婷婷啪啪免费在线电影观看 | 亚洲精品国产欧美在线观看 | 日韩精品一区二区三区视频播放 | 日本中文字幕在线看 | 久久伊人热 | 久久精品99国产精品酒店日本 | 色综合在 | 天天色宗合| av色综合| 美女久久 | 欧美人交a欧美精品 | 色中射 | 久久精品一 | 欧美激情综合五月色丁香小说 | 亚洲成av人影片在线观看 | 日韩欧美视频免费在线观看 | 国产理论在线 | 91av资源在线 | 欧美伦理一区 | 99久久精品免费看国产四区 | 久久av在线 | 在线视频1卡二卡三卡 | 亚洲五月婷婷 | 五月开心六月婷婷 | 欧美日韩一区二区三区不卡 | 久久这里只有精品视频首页 | 永久免费毛片 | 丰满少妇一级 | 日韩一二三区不卡 | 精品亚洲免a | 日韩一二区在线观看 | 精品在线看 | 99精品色| 精品久久国产一区 | 国产视频一区在线免费观看 | 五月婷婷色 | 五月天激情婷婷 | 色综合亚洲精品激情狠狠 | 激情小说久久 | 亚洲国产精品成人va在线观看 | 久久色在线观看 | .精品久久久麻豆国产精品 亚洲va欧美 | 日韩免费视频一区二区 | 日韩欧美在线视频一区二区 | 天堂成人在线 | 97在线成人 | 欧美视频网址 | 色99久久| 欧美日韩精品在线观看视频 | 日韩av片免费在线观看 | 免费观看第二部31集 | 98超碰人人 | 美女视频国产 | 精品在线观看视频 | 日韩免费电影在线观看 | 国产黄a三级三级 | 日韩欧美成 | 天天操夜夜操夜夜操 | 久久精品5 | 成人一区不卡 | av片中文 | 亚洲午夜精品一区二区三区电影院 | 成人av亚洲 | 国产免费又粗又猛又爽 | 在线免费观看不卡av | 精品国产大片 | 亚洲综合在线视频 | 亚洲国产av精品毛片鲁大师 | 日韩二区三区在线观看 | 日韩婷婷| 久久99免费视频 | 免费亚洲黄色 | 中文在线8新资源库 | 九九热在线播放 | 日韩黄色免费看 | 精品视频中文字幕 | 免费观看成年人视频 | 18国产精品白浆在线观看免费 | 一区二区精品在线 | 91成人免费在线视频 | 欧美专区国产专区 | 国产亚洲精品v | 91视频91蝌蚪 | 黄色a一级片 | 国产成人一区二区三区电影 | 四虎成人精品永久免费av | 欧美一级电影在线观看 | 日韩网站在线播放 | 日日日日 | 在线 欧美 日韩 | 一级淫片在线观看 | 99久久激情 | 久艹在线播放 | 91看片淫黄大片一级在线观看 | 狠狠色狠狠综合久久 | 国产精品a久久 | 免费看的毛片 | 日本黄色免费电影网站 | 一本之道乱码区 | av电影在线观看完整版一区二区 | 久久久久国产一区二区三区四区 | 天天看天天干天天操 | 日韩国产在线观看 | 99精品一级欧美片免费播放 | 在线观看视频一区二区三区 | www.狠狠插.com | 国产高清久久久久 | 韩国av一区二区三区在线观看 | 2018亚洲男人天堂 | 亚洲黄色免费网站 | 人人爽人人爱 | 天天插天天 | 日韩精品在线观看av | 欧美吞精 | 91视频免费看网站 | 中文一区在线观看 | 少妇视频一区 | 性色av香蕉一区二区 | 爱情影院aqdy鲁丝片二区 | 91成人久久 | 日韩欧美视频免费观看 | 国产精品久久久久久久久久久久 | 日日干天天爽 | 天天射天天搞 | 国产亚洲精品日韩在线tv黄 | 日本精品视频在线观看 | 久久久免费高清视频 | 精品国产片 | 久久午夜影院 | 国产小视频在线免费观看 | 中文字幕在线字幕中文 | 久久综合欧美精品亚洲一区 | 91麻豆精品久久久久久 | 五月天综合激情 | 欧美久久久久久久久久久久 | 欧美日韩高清一区二区 国产亚洲免费看 | 日本h视频在线观看 | 中文成人字幕 | 在线a视频 | 6080yy午夜一二三区久久 | 99热超碰在线 | 日韩二区三区在线 | 中文字幕在线观看视频一区 | 午夜视频导航 | 五月花丁香婷婷 | 亚洲在线视频播放 | 97超碰人人模人人人爽人人爱 | 亚洲手机天堂 | 久久久午夜精品理论片中文字幕 | 国产精品久久伊人 | 国产麻豆精品95视频 | 亚洲精品麻豆视频 | 日本精品xxxx | 国语精品免费视频 | 亚洲有 在线 | 国产欧美精品一区二区三区四区 | 中文字幕有码在线观看 | 天天干夜夜夜 | 在线播放国产精品 | 婷婷丁香社区 | 日韩精品久久久久久久电影竹菊 | 视频在线观看91 | av在线免费在线观看 | 丁香花中文在线免费观看 | 国产第一福利网 | 97超碰资源总站 | 久久免费视频在线 | 欧洲精品亚洲精品 | 国产一级电影免费观看 | 国产精品久久久久久久久久久久久久 | 国产一级小视频 | 精品国产乱码久久久久久浪潮 | 天堂入口网站 | 久久久久在线视频 | 在线黄av | 人人玩人人添人人澡97 | 亚洲第一区精品 | 亚洲高清视频在线观看 | 最新中文字幕在线资源 | 一区二区三区四区五区在线 | 久久久久免费精品国产小说色大师 | 久久综合精品一区 | 亚洲成人午夜在线 | 国产精品久免费的黄网站 | 久久婷婷精品 | 九九九热精品免费视频观看网站 | 精品国产精品久久一区免费式 | 免费福利片 | 欧美亚洲精品在线观看 | 在线观看日本高清mv视频 | 久久免费精品视频 | 激情综合五月 | 国产视频美女 | 亚洲欧洲国产日韩精品 | 91精品影视 | 久久在线免费观看 | 国产69精品久久久久9999apgf | 四虎影视成人精品国库在线观看 | 精品国产123| 色wwww| 国产亚洲欧美一区 | 91av视频在线播放 | 大胆欧美gogo免费视频一二区 | 草久在线 | 六月丁香综合网 | 日日夜夜天天久久 | av超碰在线| www在线观看国产 | 97av精品 | 国产视频在线观看一区 | 日本精品一二区 | av再线观看 | 欧美日韩视频在线播放 | 97色在线视频 | 中文网丁香综合网 | 成人h视频在线播放 | 国产精品涩涩屋www在线观看 | 日韩艹 | 日韩理论片在线观看 | 国产在线观看一区 | 亚洲乱码中文字幕综合 | 九九热中文字幕 | 欧美另类调教 | 在线国产视频 | 一级a性色生活片久久毛片波多野 | 日韩在线观看中文 | 久久综合九色综合欧美就去吻 | av亚洲产国偷v产偷v自拍小说 | 六月丁香激情网 | 激情五月六月婷婷 | 精品久久亚洲 | 国产在线观看中文字幕 | 久久免费视频网 | 国产你懂的在线 | 婷婷激情5月天 | 欧美专区国产专区 | 丝袜美腿亚洲 | 欧美资源 | 欧美日韩中文国产一区发布 | 国产黄a三级三级三级三级三级 | 国内精品久久久精品电影院 | 伊人婷婷网 | 亚洲国产精品第一区二区 | 久久久久久久久久久影院 | 91中文字幕在线视频 | 亚洲精品国产第一综合99久久 | 九色精品免费永久在线 | 性色av免费在线观看 | 国产xx在线 | 精品成人免费 | 人人狠狠综合久久亚洲婷 | 国产不卡精品视频 | 毛片区| 国产精品乱码一区二三区 | 国产精品麻豆免费版 | 女人魂免费观看 | 免费看网站在线 | 又黄又刺激视频 | 青青河边草观看完整版高清 | 久草网在线观看 | 国产成人一区二区在线观看 | 六月天色婷婷 | 国产黄视频在线观看 | 欧美日韩国产二区 | 日韩在线观看免费 | 精品99久久 | 高清av中文在线字幕观看1 | 久久精品视频3 | 欧美日韩性视频在线 | 日本久久电影网 | 国产精彩在线视频 | 欧美日韩首页 | 久久综合偷偷噜噜噜色 | 91av福利视频 | 国产精品亚洲人在线观看 | 久久一区91 | 久久国产精品久久久久 | 久久色中文字幕 | 99精品国产一区二区 | 综合网五月天 | 国产亚洲精品久久19p | 波多野结衣在线播放一区 | 久久av免费观看 | 精品综合久久久 | 97视频在线播放 | 在线免费观看黄色 | 国产成年免费视频 | 欧美激情亚洲综合 | 国产精品黑丝在线观看 | 1000部国产精品成人观看 | 五月激情在线 | 久久黄色免费观看 | 亚洲天堂香蕉 | 国产成人黄色在线 | 天天摸天天操天天爽 | 久久免费视频4 | 成人免费在线观看电影 | 久久综合五月天婷婷伊人 | 久久久久国产成人免费精品免费 | 麻豆国产精品一区二区三区 | 免费欧美高清视频 | 日韩免费成人 | 激情五月婷婷综合 | 欧美一级久久久久 | 精品国产免费看 | 一区二区三区四区精品视频 | 亚洲资源在线观看 | 中文字幕在线视频网站 | 成人小视频免费在线观看 | 在线观看91av | 婷婷久久综合九色综合 | 日日夜夜中文字幕 | 免费观看v片在线观看 | 国产一级电影网 | 亚洲精品成人在线 | 国产精品麻 | 热久久免费视频精品 | 91精品久久久久久久久久入口 | 日韩小视频网站 | 特级大胆西西4444www | 亚洲精品乱码久久久久久久久久 | 中文av字幕在线观看 | 天堂av网址 | 日韩一区精品 | 免费一级日韩欧美性大片 | 日韩中文字幕免费在线观看 | 91伊人久久大香线蕉蜜芽人口 | 欧美日韩在线观看一区二区三区 | 97国产大学生情侣白嫩酒店 | 美女精品网站 | 国产中文字幕国产 | 日韩精品一区二区三区外面 | 国产性xxxx| 久久精品99久久久久久2456 | 人人狠狠综合久久亚洲 | 精品一区二区亚洲 | 天天天色综合 | 日韩免费在线视频观看 | 99在线观看视频 | 麻豆视频免费在线播放 | 91日韩在线 | 欧美成人日韩 | 国产精品久久久久久久久久久杏吧 | 国产一级黄色电影 | 又黄又爽的免费高潮视频 | 亚洲精品视频在线观看免费视频 | 色国产精品 | 91桃色免费视频 | 伊人黄色网 | 免费看的毛片 | 韩国一区视频 | 久久久精品电影 | 亚洲视频电影在线 | 天天干天天拍天天操天天拍 | 综合黄色网| 夜夜澡人模人人添人人看 | 成人国产精品一区二区 | 在线va网站| 欧美日韩中文字幕视频 | 国内精品久久久久影院一蜜桃 | 欧美人操人 | 四虎免费在线观看视频 | 成人小视频在线观看免费 | 日韩午夜大片 | 久久伊人精品一区二区三区 | 国产精品一区二区久久精品爱微奶 | a特级毛片 | 麻豆一二三精选视频 | 美女网站久久 | 成人国产一区二区 | 国产在线观看中文字幕 | 久久久久久久电影 | 免费av 在线 | 国产精品久久久久永久免费看 | 日日日干 | 欧美在线视频精品 | 精品久久久久久久久久久久久久久久 | 国产精品一区在线观看你懂的 | 欧美精品免费在线观看 | a天堂中文在线 | 国产精品一区二区美女视频免费看 | 国产成人久久精品77777 | 久久你懂的 | 国产中年夫妇高潮精品视频 | 国产破处在线视频 | 丁香九月婷婷 | 色中文字幕在线观看 | 狠狠干天天射 | 久久欧美精品 | 久久综合九色综合久久久精品综合 | 日韩毛片在线免费观看 | 一区二区三区久久精品 | 片黄色毛片黄色毛片 | 天天干人人 | 精品视频99| 欧美日韩国产精品一区二区 | 91正在播放 | 久久久久久免费视频 | 久草爱视频| 国产精品成人自产拍在线观看 | 草樱av| 亚洲精品综合久久 | 国产人成一区二区三区影院 | 欧美精品久久久久性色 | 中文永久字幕 | 日韩成人在线一区二区 | 在线观看久草 | 亚洲黄色小说网址 | 欧美日韩国产mv | 五月综合在线观看 | 久久精品一区 | 亚洲乱码精品久久久 | 久久精品导航 | 天天插天天色 | 欧美日韩亚洲第一 | 99久久精品免费看国产一区二区三区 | 亚洲丝袜中文 | 久久免费福利视频 | 亚洲综合一区二区精品导航 | 久久99国产一区二区三区 | 日韩免费视频在线观看 | 日韩在线三区 | 久久精品福利视频 | 欧美日韩中 | 国产精品黄色在线观看 | 中文字幕999 | 91av九色| 六月婷婷久香在线视频 | 久热av | 最近最新中文字幕视频 | 在线观看91精品视频 | 五月开心激情网 | www.午夜色.com | 国产精品美乳一区二区免费 | 在线观看视频免费大全 | 中文字幕一区2区3区 | 福利一区二区在线 | 国产高清成人 | 久久99久久99精品中文字幕 | 日韩高清无线码2023 | 成人在线视频免费看 | 亚洲精品视频一二三 | 国产日韩在线视频 | 日韩精品免费 | 福利视频网站 | 最近中文字幕大全中文字幕免费 | 五月激情天 | 精品视频在线免费观看 | 国产成人免费 | 黄色小网站在线 | 在线免费观看黄色大片 | 日本久久久影视 | 日韩精品免费在线播放 | 免费看成人a | 久久 一区 | 日韩精品无 | 视频在线精品 | 免费看片黄色 | 一区二区三区四区影院 | 日韩精品国产一区 | 婷婷精品 | 操天天操| 久久久久在线 | 久久精品99 | 高清免费在线视频 | 激情久久久久久久久久久久久久久久 | 免费亚洲黄色 | 色综合天天色 | 草 免费视频 | 精品国产乱码久久久久久浪潮 | 高清久久久 | 国产精品久久久久婷婷二区次 | 中文永久免费观看 | 麻花豆传媒mv在线观看网站 | 中文字幕永久在线 | 国产精品嫩草影院99网站 | 国产精品久久久久久a | av三级在线免费观看 | 99热精品国产一区二区在线观看 | 日韩三级不卡 | 亚洲视频,欧洲视频 | 日韩视频一区二区三区在线播放免费观看 | www.色五月.com | 91精品免费 | 亚洲精品天天 | 毛片网在线观看 | 国产三级香港三韩国三级 | 欧美男同网站 | 999久久 | 福利一区在线 | 午夜丰满寂寞少妇精品 | 9在线观看免费高清完整版 玖玖爱免费视频 | 欧洲精品在线视频 | 黄色福利视频网站 | 中文字幕在线观看视频一区 | 97国产一区| 国产又粗又长又硬免费视频 | 久久久久女人精品毛片九一 | 国产精品99久久久精品免费观看 | 久久伊99综合婷婷久久伊 | 日日摸日日 | 91精品视频免费在线观看 | 国产剧情一区二区在线观看 | 麻豆免费在线视频 | 国产精在线 | 人人爽人人看 | 黄免费在线观看 | 97精品在线视频 | 亚洲欧洲精品一区二区 | 色欧美88888久久久久久影院 | 欧美aa在线 | 久草在线在线精品观看 | av中文字幕不卡 | 亚洲片在线 | 91av视频免费在线观看 | 在线观看视频在线 | 97国产精品免费 | 麻豆视频大全 | 久久久久久久久久久福利 | 国产免费又爽又刺激在线观看 | 成片视频在线观看 | 日日夜夜操av | 亚洲欧洲一区二区在线观看 | 精品资源在线 | 国产蜜臀av | 成人中文字幕在线观看 | 97狠狠干| 国产剧情一区在线 | 免费精品视频 | 日韩av视屏在线观看 | 国产不卡网站 | 国产精品色婷婷视频 | 天天天色 | 色偷偷888欧美精品久久久 | 国产成人一区二区三区在线观看 | 欧美日韩精品区 | 亚洲一级电影在线观看 | 美女在线免费视频 | 久久久久久久久电影 | 日日麻批40分钟视频免费观看 | 久久精品99国产精品 | 亚洲一区二区三区毛片 | 日韩在线视频看看 | 亚洲成a人片综合在线 | www久| 国产成人亚洲在线观看 | 久久久 激情| 国产一区二区三区四区在线 | 在线小视频 | 久久精视频 | 日韩欧美高清一区二区三区 | 精品一区久久 | av大片免费 | 国产伦精品一区二区三区免费 | 国产精品黄 | 国产成人久久av977小说 | 成人综合日日夜夜 | 国产高清绿奴videos | 婷婷色六月天 | 亚洲天堂网在线播放 | 欧美精品一区二区免费 | 免费观看性生交 | 久久久综合电影 | 夜夜天天干 | 久久99精品国产91久久来源 | 久久五月婷婷丁香社区 | 91插插插网站 | 人人干人人添 | 国产高清亚洲 | 日韩欧美视频在线免费观看 | 国产区在线视频 | 成 人 黄 色 视频 免费观看 | 国产视频精品网 | 日本不卡一区二区三区在线观看 | 中文字幕av在线免费 | 91资源在线观看 |