日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Python基于React-Dropzone實現(xiàn)上傳組件的示例代碼

瀏覽:153日期:2022-07-29 08:14:40
目錄實例演示1. axios上傳普通文件:2. 大文件導入:結語

這次我要講述的是在React-Flask框架上開發(fā)上傳組件的技巧。我目前主要以React開發(fā)前端,在這個過程中認識到了許多有趣的前端UI框架——React-Bootstrap、Ant Design、Material UI、Bulma等。而比較流行的上傳組件也不少,而目前用戶比較多的是jQuery-File-Upload和Dropzone,而成長速度快的新晉有Uppy和filepond。比較惋惜的是Fine-Uploader的作者自2018年后就決定不再維護了,原因作為后來者的我就不多過問了,但請各位尊重每一位開源作者的勞動成果。

這里我選擇React-Dropzone,原因如下:

基于React開發(fā),契合度高 網(wǎng)上推薦度高,連Material UI都用他開發(fā)上傳組件 主要以 Drag 和 Drop 為主,但是對于傳輸邏輯可以由開發(fā)者自行設計。例如嘗試用socket-io來傳輸file chunks。對于node全棧估計可行,但是我這里使用的是Flask,需要將Blob轉(zhuǎn)ArrayBuffer。但是如何將其在Python中讀寫,我就沒進行下去了。 實例演示1. axios上傳普通文件:

通過yarn將react-dropzone和引入:

yarn add react-dropzone axios

前端js如下(如有缺失,請自行修改):

import React, { useState, useCallback, useEffect,} from ’react’;import {useDropzone} from ’react-dropzone’;import './dropzone.styles.css'import InfiniteScroll from ’react-infinite-scroller’;import { List, message, // Avatar, Spin,} from ’antd’;import axios from ’axios’;/*** 計算文件大小* @param {*} bytes * @param {*} decimals * @returns */function formatBytes(bytes, decimals = 2) { if (bytes === 0) return ’0 Bytes’; const k = 1024; const dm = decimals < 0 ? 0 : decimals; const sizes = [’Bytes’, ’KB’, ’MB’, ’GB’, ’TB’, ’PB’, ’EB’, ’ZB’, ’YB’]; const i = Math.floor(Math.log(bytes) / Math.log(k)); return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ’ ’ + sizes[i];}/*** Dropzone 上傳文件* @param {*} props * @returns */function DropzoneUpload(props) { const [files, setFiles] = useState([]) const [loading, setLoading] = useState(false); const [hasMore, setHasMore] = useState(true); const onDrop = useCallback(acceptedFiles => {setLoading(true);const formData = new FormData();smallFiles.forEach(file => { formData.append('files', file);});axios({ method: ’POST’, url: ’/api/files/multiplefiles’, data: formData, headers: {'Content-Type': 'multipart/form-data', }})then(resp => { addFiles(acceptedFiles); setLoading(false);}); }, [files]); // Dropzone setting const { getRootProps, getInputProps } = useDropzone({multiple:true,onDrop, }); // 刪除附件 const removeFile = file => {const newFiles = [...files]newFiles.splice(newFiles.indexOf(file), 1)setFiles(newFiles) } useEffect(() => {// init uploader filessetFiles([]) },[]) return (<section className='container'><div {...getRootProps({className: ’dropzone’})}> <input {...getInputProps()} /> <p>拖動文件或點擊選擇文件😊</p></div><div className='demo-infinite-container'> <InfiniteScrollinitialLoad={false}pageStart={0}loadMore={handleInfiniteOnLoad}hasMore={!loading && hasMore}useWindow= {false} ><List dataSource={files} renderItem={item=> (<List.Item actions={[// <a key='list-loadmore-edit'>編輯</a>, <a key='list-loadmore-delete' onClick={removeFile}>刪除</a> ]} // extra={// } key={item.path}> <List.Item.Meta avatar={ <> {!!item.type && [’image/gif’, ’image/jpeg’, ’image/png’].includes(item.type) &&<img width={100} alt=’logo’ src={item.preview}/> } </>}title={item.path}description={formatBytes(item.size)} /></List.Item> )}> {loading && hasMore && (<div className='demo-loading-container'> <Spin /></div> )}</List> </InfiniteScroll></div></section> );}

flask代碼:

def multiplefiles():if ’files’ not in request.files: return jsonify({’message’: ’沒有文件!’}), 200files = request.files.getlist(’files’)for file in files: if file:# 通過拼音解決secure_filename中文問題filename = secure_filename(’’.join(lazy_pinyin(file.filename))Path(UPLOAD_FOLDER + ’/’ + file_info[’dir_path’]).mkdir(parents=True, exist_ok=True)file.save(os.path.join(UPLOAD_FOLDER + ’/’ + file_info[’dir_path’], filename))return jsonify({’message’: ’保存成功!!’})2. 大文件導入:

通過file.slice()方法生成文件的chunks。不要用Promise.all容易產(chǎn)生非順序型的請求,導致文件損壞。

js代碼:

const promiseArray = largeFiles.map(file => new Promise((resolve, reject) => { const chunkSize = CHUNK_SIZE; const chunks = Math.ceil(file.size / chunkSize); let chunk = 0; let chunkArray = new Array(); while (chunk <= chunks) {let offset = chunk * chunkSize;let slice = file.slice(offset, offset+chunkSize)chunkArray.push([slice, offset])++chunk; } const chunkUploadPromises = (slice, offset) => {const largeFileData = new FormData();largeFileData.append(’largeFileData’, slice)return new Promise((resolve, reject) => { axios({method: ’POST’,url: ’/api/files/largefile’,data: largeFileData,headers: { 'Content-Type': 'multipart/form-data'} }) .then(resp => {console.log(resp);resolve(resp); }) .catch(err => {reject(err); })}) }; chunkArray.reduce( (previousPromise, [nextChunk, nextOffset]) => {return previousPromise.then(() => { return chunkUploadPromises(nextChunk, nextOffset);}); }, Promise.resolve()); resolve();}))

flask代碼:

filename = secure_filename(’’.join(lazy_pinyin(filename)))Path(UPLOAD_FOLDER + ’/’ + file_info[’dir_path’]).mkdir(parents=True, exist_ok=True)save_path = os.path.join(UPLOAD_FOLDER + ’/’ + file_info[’dir_path’], filename)# rm file if existsif offset == 0 and save_path.exists(filename): os.remove(filename)try: with open(save_path, ’ab’) as f:f.seek(offset)f.write(file.stream.read())print('time: '+ str(datetime.now())+' offset: ' + str(offset))except OSError: return jsonify({’Could not write to file’}), 500結語

文件傳輸一直都是HTTP的痛點,尤其是大文件傳輸。最好的方式是自己做個Client,通過FTP和FTPS的協(xié)議進行傳輸。第二種來自于大廠很中心化的方法,通過文件的checksum來確定文件是否已經(jīng)上傳了,來營造秒傳的效果。第三種來自去中心化的Bittorrent的方法每一個用戶做文件種子,提供文件傳輸?shù)妮o助,目前國內(nèi)并沒有普及使用。

到此這篇關于Python基于React-Dropzone實現(xiàn)上傳組件的示例代碼的文章就介紹到這了,更多相關Python React-Dropzone上傳組件內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持好吧啦網(wǎng)!

標簽: Python 編程
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
深夜视频一区二区| 给我免费播放日韩视频| 99久久亚洲精品蜜臀| 麻豆国产在线| 日韩av一级| 国产精品黑丝在线播放| 精品视频一区二区三区四区五区| 国产欧美日韩精品一区二区免费 | 国产免费久久| 国产精品观看| 国产91在线精品| 麻豆精品蜜桃| 999在线观看精品免费不卡网站| 欧美日韩视频一区二区三区| 国产毛片久久| 日韩精品国产欧美| 国产伦一区二区三区| 欧美亚洲二区| 久久一区欧美| 91看片一区| 一区在线免费| 日韩一区二区三区四区五区| 欧美精品影院| 三上亚洲一区二区| 欧美日韩精品免费观看视频完整| 一区二区精彩视频| 欧美精品三级在线| 鲁鲁在线中文| 麻豆91精品| 国产精品毛片视频| 久久精品成人| 蜜桃视频免费观看一区| 日欧美一区二区| 精品一区二区三区亚洲| av亚洲在线观看| 日本不卡一二三区黄网| 国产成人调教视频在线观看| 一区二区小说| 日本视频一区二区| 久久青青视频| 久久亚洲欧美| 久久精品一区| 久久aⅴ国产紧身牛仔裤| 欧美日韩精品一区二区三区视频| 精品久久久中文字幕| 欧美va天堂| 日韩欧美中文字幕电影| 久久99高清| 亚洲深夜影院| 久久久国产精品网站| 在线精品小视频| 国产情侣久久| 黄色日韩在线| 精品一区二区男人吃奶| 丝袜美腿一区二区三区| 精品高清久久| 午夜在线视频观看日韩17c| 麻豆免费精品视频| 亚洲一区日韩| 久久久久久自在自线| 亚洲日产av中文字幕| 97国产成人高清在线观看| 亚洲影院天堂中文av色| 国产一区二区三区亚洲综合| 蜜臀久久99精品久久久久久9| 红杏一区二区三区| 亚洲免费一区三区| 久久高清免费| 国产成人精品一区二区三区视频| 免费观看在线综合色| 亚洲啊v在线| 国产精品麻豆成人av电影艾秋| 日韩视频中文| 日韩精品永久网址| 欧美日韩 国产精品| 9色国产精品| 国产精品99一区二区三区| 亚洲精品乱码久久久久久蜜桃麻豆| 日韩欧美一区二区三区免费观看| 国产欧美日韩综合一区在线播放| 国产精品呻吟| 日韩精品欧美| 国产96在线亚洲| 久久国产精品免费精品3p| 久久不射中文字幕| 特黄特色欧美大片| 免费一级欧美在线观看视频| 亚洲三级在线| 国产亚洲毛片| 欧美日韩在线播放视频| 成人午夜网址| 国产精品久久久久久久免费软件| 在线亚洲精品| 黄色aa久久| 精品三级在线| 国产精品麻豆成人av电影艾秋| 中文字幕日本一区二区| 精品久久久亚洲| 欧美视频精品全部免费观看| 先锋影音国产一区| 99国产成+人+综合+亚洲欧美| 亚洲精品一级二级| 久久精品国产久精国产| 国产精品久久久久久久久久白浆| 蜜桃av一区| 老司机精品久久| 红桃视频欧美| 日韩欧美少妇| 亚洲人成在线网站| 日韩久久一区二区三区| аⅴ资源天堂资源库在线| 久久久91麻豆精品国产一区| 国产精品久久久久久久免费软件| 69堂免费精品视频在线播放| 久久国产精品久久久久久电车 | 国产美女亚洲精品7777| 欧美天堂一区| 国产欧美自拍| 日韩一区二区三区高清在线观看| 久久国产精品久久w女人spa| 亚洲作爱视频| 久久不见久久见免费视频7| 最新国产精品久久久| 欧美在线首页| 福利一区在线| 伊人久久大香线蕉av超碰演员| 亚洲一区二区小说| 国产精品白丝av嫩草影院| 国产自产自拍视频在线观看| 香蕉精品久久| 日韩在线黄色| 四虎国产精品免费观看| 精品中文字幕一区二区三区av| 国产亚洲在线观看| 国产亚洲精品精品国产亚洲综合| 精品免费av| 极品日韩av| 日韩动漫一区| 日韩一区二区在线免费| 天堂va蜜桃一区二区三区| 国产激情一区| 亚洲二区在线| 日韩欧美激情| 福利一区视频| 免费日韩精品中文字幕视频在线| 国产精品一区二区三区www| 日韩一区欧美| 亚洲精品乱码日韩| 精品午夜av| 夜夜嗨av一区二区三区网站四季av| 日韩av二区在线播放| 蜜臀国产一区| 97精品久久| 成人羞羞在线观看网站| 中文字幕亚洲影视| 成人在线丰满少妇av| 久久亚洲精品伦理| 国产一区调教| 老牛影视一区二区三区| 精品国产亚洲一区二区三区| 亚洲自拍另类| 欧美久久香蕉| 成人日韩精品| 综合一区二区三区| 国产欧美高清| 久久国产亚洲| 国产精品一区二区三区av麻| 在线天堂中文资源最新版| 亚洲+小说+欧美+激情+另类| 国产精品99久久精品| 综合亚洲自拍| 99久久激情| 欧美国产精品| 亚洲精品一级| 99久久精品国产亚洲精品| 国产精品亚洲人成在99www| 亚洲视频播放| 国产91在线播放精品| 日韩国产在线观看| 亚洲1234区| 国产精品久久久久久久久免费高清 | 成人在线视频免费看| 中文无码久久精品| 亚洲成人精品| 久久精品一区二区三区中文字幕| 老牛国产精品一区的观看方式| 亚洲国产福利| 欧美国产专区| 日韩精品免费一区二区夜夜嗨| 欧美日韩国产在线观看网站 | 日韩欧美字幕| 国产精品啊v在线| 香蕉久久久久久| 亚洲综合电影一区二区三区| 亚洲风情在线资源| 麻豆国产一区| 国产日产一区| 亚洲人成精品久久久| 在线一区电影| 亚洲高清毛片|