婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > Python基于React-Dropzone實(shí)現(xiàn)上傳組件的示例代碼

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

熱門標(biāo)簽:宿遷星美防封電銷卡 ai電話機(jī)器人哪里好 湛江智能外呼系統(tǒng)廠家 外呼并發(fā)線路 西藏房產(chǎn)智能外呼系統(tǒng)要多少錢 ai電銷機(jī)器人源碼 長沙高頻外呼系統(tǒng)原理是什么 地圖標(biāo)注審核表 百度地圖標(biāo)注沒有了

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

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

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

實(shí)例演示

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 files
        setFiles([])
    },[])

    return (
        section className="container">
        div {...getRootProps({className: 'dropzone'})}>
            input {...getInputProps()} />
            p>拖動文件或點(diǎn)擊選擇文件😊/p>
        /div>
        
        div className="demo-infinite-container">
            InfiniteScroll
                initialLoad={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': '沒有文件!'}), 200
files = 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. 大文件導(dǎo)入:

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

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 exists
if 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

結(jié)語

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

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

您可能感興趣的文章:
  • python 實(shí)現(xiàn)上傳圖片并預(yù)覽的3種方法(推薦)
  • 用Python實(shí)現(xiàn)一個簡單的能夠上傳下載的HTTP服務(wù)器
  • python實(shí)現(xiàn)上傳下載文件功能
  • Python selenium文件上傳方法匯總
  • python實(shí)現(xiàn)的簡單FTP上傳下載文件實(shí)例
  • Python+django實(shí)現(xiàn)文件上傳
  • python3 flask實(shí)現(xiàn)文件上傳功能
  • Python基于FTP模塊實(shí)現(xiàn)ftp文件上傳操作示例

標(biāo)簽:漯河 林芝 寧夏 大同 盤錦 南平 普洱 海南

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Python基于React-Dropzone實(shí)現(xiàn)上傳組件的示例代碼》,本文關(guān)鍵詞  Python,基于,React-Dropzone,實(shí)現(xiàn),;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《Python基于React-Dropzone實(shí)現(xiàn)上傳組件的示例代碼》相關(guān)的同類信息!
  • 本頁收集關(guān)于Python基于React-Dropzone實(shí)現(xiàn)上傳組件的示例代碼的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    韩国成人精品a∨在线观看| 国产精品 欧美精品| 亚洲国产日韩一区二区| 国产成a人亚洲| 日韩精品综合一本久道在线视频| 一区二区三区免费网站| 成人国产精品免费观看动漫| 精品国产乱码久久久久久老虎 | 国产精品每日更新| 麻豆精品视频在线观看视频| 日韩欧美电影在线| 成人av电影在线| 亚洲日本中文字幕区| 色婷婷亚洲婷婷| 午夜精品久久久久久久99樱桃| 欧美一级在线视频| 国产精品伊人色| 一区二区三区精品视频| 69堂国产成人免费视频| 成人免费看视频| 亚洲一区中文日韩| 亚洲欧洲三级电影| 欧美亚洲国产一区在线观看网站| 亚洲欧美另类小说视频| 国产亚洲欧美色| 欧美视频第二页| 成人av电影在线| 久久er精品视频| 肉丝袜脚交视频一区二区| 欧美—级在线免费片| 欧美成人女星排名| 欧美一区二区三区的| 91亚洲精品一区二区乱码| 六月婷婷色综合| 久久av老司机精品网站导航| 婷婷国产在线综合| 性欧美大战久久久久久久久| 亚洲欧美二区三区| 最新高清无码专区| 日韩一区有码在线| 亚洲免费看黄网站| 亚洲三级在线免费| 国产精品每日更新在线播放网址 | 欧美一区二区三区四区高清| 欧美日韩一区二区在线观看视频 | 亚洲成av人**亚洲成av**| 一区二区久久久久久| 日韩精品福利网| 成人免费视频网站在线观看| 99免费精品在线| 久久你懂得1024| 亚洲人成网站在线| 五月天激情小说综合| 懂色av一区二区三区免费观看| 成人午夜免费视频| 欧美丰满一区二区免费视频| 精品国产91亚洲一区二区三区婷婷| 国产日韩欧美a| 天堂精品中文字幕在线| 激情综合网av| 欧美视频在线不卡| 欧美精品一区视频| 一区二区三区影院| 国产91精品入口| 精品国产百合女同互慰| 久久老女人爱爱| 亚洲黄色av一区| 91美女在线看| 国产精品污网站| 粉嫩av一区二区三区粉嫩 | 欧美日韩极品在线观看一区| 国产精品美日韩| 国产美女一区二区| 国产女人18水真多18精品一级做| 亚洲一区二区四区蜜桃| 91视频国产资源| 亚洲一区中文日韩| 欧美群妇大交群中文字幕| 一区二区三区中文字幕电影| 激情久久久久久久久久久久久久久久| 一区二区在线观看免费| 成人性视频网站| 精品日韩在线观看| 国产一区二区三区精品视频| 1000精品久久久久久久久| 韩国欧美国产1区| 国产区在线观看成人精品 | 国产尤物一区二区| 国产日韩一级二级三级| 色婷婷亚洲婷婷| 美国十次了思思久久精品导航| 欧美成人一区二区三区在线观看 | 91精品国产综合久久福利| 首页综合国产亚洲丝袜| 日韩免费性生活视频播放| 成人免费看的视频| 日本在线不卡一区| 中文字幕精品一区| 日韩三级电影网址| 欧美综合一区二区三区| 国产伦精品一区二区三区免费| 国产精品久久久久影院色老大| 欧美私人免费视频| 成人av午夜电影| 国产高清不卡一区| 蜜桃在线一区二区三区| 丝袜脚交一区二区| 首页综合国产亚洲丝袜| 丝瓜av网站精品一区二区| 国产精品久99| 国产网红主播福利一区二区| 欧美精选一区二区| 在线精品视频免费播放| www.日本不卡| 成人性生交大片免费看中文网站| 狠狠色狠狠色综合| 日本一区二区三区久久久久久久久不 | 一区二区三区四区乱视频| 欧美xingq一区二区| 国产精品1区2区3区在线观看| 亚洲综合成人在线| 亚洲人妖av一区二区| 制服.丝袜.亚洲.另类.中文| 91影院在线观看| 久久国内精品自在自线400部| 性感美女极品91精品| 亚洲成人av一区| 日韩高清在线一区| 青青草国产精品亚洲专区无| 国内精品伊人久久久久av影院 | 久久色.com| 亚洲综合在线电影| 精品福利一区二区三区| 亚洲日本va午夜在线电影| 亚洲五月六月丁香激情| 午夜精品久久久久久久| 国产精品一区二区久久精品爱涩 | 天天综合天天做天天综合| 国产凹凸在线观看一区二区| 成人激情文学综合网| 欧美日产在线观看| 亚洲另类色综合网站| 国产一区三区三区| 精品日韩欧美在线| 午夜精品视频在线观看| 国产在线播精品第三| 99久久er热在这里只有精品66| 欧美一级理论片| 亚洲aaa精品| 欧美日韩一级大片网址| 成人欧美一区二区三区在线播放| 国产经典欧美精品| 日韩午夜激情视频| 国产在线日韩欧美| |精品福利一区二区三区| 成人动漫一区二区| 亚洲人精品午夜| 欧美精品少妇一区二区三区| 性欧美大战久久久久久久久| 欧美人牲a欧美精品| 免费精品视频在线| 国产精品污污网站在线观看| 成人高清免费在线播放| 亚洲综合丝袜美腿| 久久久久国色av免费看影院| 国产a精品视频| 亚洲国产精品久久久久婷婷884 | 国产精品 欧美精品| 国产欧美日韩在线看| 99综合影院在线| 婷婷开心久久网| 2020国产成人综合网| 色哟哟国产精品免费观看| 国内外成人在线| 亚洲一区二区三区四区在线 | 激情五月播播久久久精品| 欧美大片日本大片免费观看| 成人自拍视频在线| 日本亚洲一区二区| 亚洲免费观看在线观看| 久久影院午夜论| 精品久久99ma| 精品国产91洋老外米糕| 67194成人在线观看| 成人黄色免费短视频| 久久精品久久久精品美女| 日韩高清在线观看| 天堂成人免费av电影一区| 亚洲动漫第一页| 日韩成人精品视频| 日韩精品亚洲专区| 久久精品二区亚洲w码| 奇米精品一区二区三区在线观看 | 日韩一级高清毛片| 日韩一区二区高清| 国产精品久久久久久户外露出| 国产日韩欧美精品在线| 中文字幕五月欧美| 国产日本欧美一区二区| 日韩精品一区二区三区视频播放|