PHP 之分片上传

一、效果图

二、前端

复制代码
<template>
    <div>
        <el-upload ref="uploadRef" 
        :show-file-list="false" 
        class="upload" 
        :auto-upload="false" 
        :on-change="handleChange"
        :limit="1" 
        :http-request="test"
        :on-exceed="test"
        >
            <el-button type="primary">文件上传</el-button>
        </el-upload>

        <el-progress :percentage="percent" :stroke-width="10" :status="status" />
    </div>
</template>

<script setup>
    import {
        getCurrentInstance,
        ref
    } from 'vue';
    
    const {proxy} = getCurrentInstance()
    const uploadRef = ref()
    
    const percent = ref(0)
    const status = ref('')

    const handleChange = (uploadFile, uploadFiles) => {
        console.log(uploadFile, uploadFiles, uploadFile.raw instanceof File)
        status.value = ''
        percent.value = 0
        
        const file = uploadFile.raw;
        const CHUNK_SIZE = 1024 * 1024 * 5; // 分片大小,例如5MB
        const CHUNKS = Math.ceil(file.size / CHUNK_SIZE); // 总分片数
        let uploaded = 0; // 已上传的分片数
        console.log(CHUNKS);

        const uploadChunk = async (chunkIndex) => {
            const start = chunkIndex * CHUNK_SIZE;
            const end = (chunkIndex + 1) * CHUNK_SIZE;
            const chunk = file.slice(start, end);
            const formData = new FormData();
            formData.append('file', chunk); // 注意:服务器端需要处理这种分片上传的情况
            formData.append('chunkIndex', chunkIndex); // 添加分片索引信息
            formData.append('totalChunks', CHUNKS); // 添加总分片数信息
            formData.append('fileName', file.name); // 可选,添加文件名信息
            
            console.log(chunk, formData)
            let res = await proxy.$http.post('upload.php', formData)
            console.log(res)
            if(res.code == 0) {
                uploaded++;
                updateProgress();
                //更新进度条
                if(chunkIndex + 1 == CHUNKS) {
                    console.log('上传成功 ==> ', res.fileInfo)
                    // alert(res.msg)
                    status.value = 'success';
                    uploadRef.value.clearFiles()
                }else{
                    uploadChunk(uploaded)
                }
            }else{
                console.log(res.msg);
            }
        }
        
        const updateProgress = () => {
            const percentVal = parseInt(parseFloat(uploaded / CHUNKS).toFixed(2) * 100)
            percent.value = percentVal
        }
        
        uploadChunk(0);
    }
    
    const test = (file) => {
        console.log(file)
    }
</script>

<style>
</style>

三、后端

复制代码
<?php

// var_dump($_FILES);
$chunkIndex = intval($_POST['chunkIndex']);
$fileName = trim($_POST['fileName']);
$totalChunks = intval($_POST['totalChunks']);

$ext = pathinfo($fileName, PATHINFO_EXTENSION);
$temp_name = md5($fileName).'_{0}.txt';
$uploadFile = str_replace('{0}', $chunkIndex, $temp_name);
if(move_uploaded_file($_FILES['file']['tmp_name'], $uploadFile)) {
    //合并文件
    $f = md5($fileName).'.'.$ext;
    if($chunkIndex + 1 == $totalChunks){
        $filesize = 0;
        for($i = 0; $i < $totalChunks; $i++){
            $file = str_replace('{0}', $i, $temp_name);
            if(file_exists($file)){
                //合并文件
                $content = file_get_contents($file);
                $filesize += strlen($content);
                file_put_contents($f, $content, FILE_APPEND);
                unlink($file);
            }
        }
        $fileInfo = array(
            'filename' => $fileName,
            'filesize' => $filesize,
            'path' => $f,
            'fileext' => $ext,
        );
        die(json_encode(array('code' => 0, 'msg' => '上传成功', 'fileInfo' => $fileInfo)));
    }
    die(json_encode(array('code' => 0, 'msg' => '上传成功')));
}else{
    die(json_encode(array('code' => 1, 'msg' => '上传失败')));
}
相关推荐
Xin_ye1008612 小时前
C# 零基础到精通教程 - 第十八章:部署与发布——让应用上线
开发语言·c#
思麟呀13 小时前
C++11并发编程:call_once一次性执行+atomic原子类型+CAS无锁编程+自旋锁
linux·开发语言·jvm·c++·windows
码不停蹄的玄黓13 小时前
Java 生产者-消费者模型详解
java·开发语言·python
爱讲故事的13 小时前
操作系统第一讲复习:为什么学习操作系统,以及操作系统到底在做什么?
linux·开发语言·windows·学习·ubuntu·c#
笨蛋不要掉眼泪13 小时前
Java并发编程:Executors框架类深度解析
java·开发语言·并发
_童年的回忆_14 小时前
【php】在linux下PHP安装amqp扩展
linux·开发语言·php
郑洁文14 小时前
基于Python的网络入侵检测系统
网络·python·php
AIMath~14 小时前
python中的uv命令揭秘
开发语言·python·uv
弹简特14 小时前
【零基础学Python】06-Python模块和包、异常处理、文件常用操作
开发语言·python
x***r15114 小时前
Postman-win64-7.2.2-Setup安装步骤详解(附API接口测试与参数配置教程)
开发语言·lua