虚拟主机 如何上传大于100M的文件 php网站程序

Mr.zhuMr.zhu2025-09-10 09:30:09来源:数学应用 (www.shuxueyingyong.com)阅读:40

问题

  1. 虚拟主机上传文件大小限制100m,

  2. 有时会遇到非常大的文件上传,上传过程中耗时非常久,

  3. 可能服务器的限制设置了上传文件尺寸,返回“413 request entity too large”

整体逻辑

  1. 前端:上传文件时,进行文件分片;发起请求时,带上第几次分片上传、总片数。

  2. 后端:按照分片进行文件保存,当上传完最后一片数据时,进行文件合并,并删除分片文


示例代码 下载:  http://downinfo.myhostadmin.net/upload.zip


只是演示功能,生产环境需要加强上传过滤


前端 upload.html 

<html>
<head>
<meta charset="utf-8">
<title>分片上传</title>
<script src="http://code.jquery.com/jquery-1.11.1.min.js"></script>
</head>
<body>
<form method="post" id="myForm" enctype="multipart/form-data">
    <input type="file" id="file" name="file">
    <input type="submit" id="submit" value="上传">
</form>


<script type="text/javascript">
    $('#submit').on('click', function(e) {
        // 阻止默认表单提交
        e.preventDefault();

        // 获取属性
        var myfile = $('#file')[0].files[0];
        var ext = myfile.name.split('.').pop();
        // 判断ext是否为视频
        var is_video = ['mp4', 'avi', 'rmvb', 'mkv'].indexOf(ext) > -1;
        var fileId = getFileIdentifier(myfile);
        // 数据切片
        var chunks = fileSlice(myfile, is_video);
        console.log(chunks)
        // 发送分割数据段
        sendChunk(fileId, chunks);
    })
    function getFileIdentifier(file){
     // 获取文件标识符
     return file.name
     //return file.size + file.name;
   }
   function fileSlice(file, is_video) {
     // 切片不宜过大,过大需要 nginx 以及 php 做相应配置
        var chunkSize =  1024 * 1024 * 4;  //切片大小控制
        // 1.初始化数据
        var totalSize = file.size;
        var start = 0;
        var end = start + chunkSize;
        var chunks = [];
        // 2.使用bolb提供的slice方法切片
        while (start < totalSize) {
            if (is_video) {
                console.log('视频')
                var chunk = file.slice(start, end, 'video/mp4');
            } else {
                console.log('图片')
                var chunk = file.slice(start, end);
            }
            chunks.push(chunk);
            start = end;
            end += chunkSize;
        }
        // 3.返回切片组chunk[]
        return chunks;
    }
    function sendChunk(id, chunks){
        // 逐个提交
        // 用于保证ajax发送完毕
        var task = [];
        var totalPage=0;
        var i=0;
        totalPage=chunks.length-1;
        var fileExt = id.substr(id.lastIndexOf('.') + 1);

        chunks.forEach(function(chunk, index){
            var formData = new FormData();
            formData.append('file',chunk);
            formData.append("fileName",id);
            formData.append("totalPage",totalPage);
            formData.append("page",index);
            $.ajax({
                type: "POST",
                url: 'upload.php',
                data: formData,
                contentType: false,
                processData: false,
                dataType:"json",
                async:false,
                success: function(data){
                    // 移除已完成任务
                    task.pop();
                    if (data['status']==200){
                        console.log(data['downUrl']);
                        alert(data['downUrl']); //返回上传文件路径
                    }
                }
                })
            task.push('file Working');
        })
    }
</script>
</body>
</html>

后端 upload.php

<?php
if (empty($_POST)) {
    $res = ['status' => 500];
    echo json_encode($res);
    exit;
}
// 创建上传目录
if(!is_dir('upload')){
   mkdir('upload', 0777);
 }
// 创建上传缓存目录
if(!is_dir('tmp')){
   mkdir('upload', 0777);
 }
 
$fileName          = isset($_POST['fileName'])?$_POST['fileName']:'';
$page              = isset($_POST['page'])?$_POST['page']:'';
$totalPage         = isset($_POST['totalPage'])?$_POST['totalPage']:'';

$fileTmpName       = isset($_FILES['file'])?$_FILES['file']['tmp_name']:'';

$status  = 206;
$downUrl = '';



if ($fileName== ''|| $page == '' || $totalPage == '' || $fileTmpName == '') {
    $res = ['status' => 500];
    echo json_encode($res);
    exit();
}




// 上传文件要保存的路径
$fname = sprintf('./tmp/%s-%s', $fileName, $page);
$data  = file_get_contents($fileTmpName);
file_put_contents($fname, $data);

// 整合分片文件
//if ($save) {
if ($totalPage ==$page) {
    $uploadFileName = sprintf('./upload/%s%s', time(),$fileName); 
    $status         = 200;
    // 合并文件,删除分片文件  
    for ($i = 0; $i<=$totalPage; $i++) {
        $tmp =  sprintf('./tmp/%s-%s', $fileName, $i);
        $data = file_get_contents($tmp);
        file_put_contents($uploadFileName, $data, FILE_APPEND);
        @unlink($tmp);
    }

    $dir = trim(dirname($_SERVER['PHP_SELF']), '/');
    if ($dir!='') {
      $dir .= '/';
    }

    $downUrl = sprintf('%s://%s/%s%s', $_SERVER['REQUEST_SCHEME'], $_SERVER['HTTP_HOST'], $dir,trim($uploadFileName, './'));
    $res = ['status' => $status,'downUrl' => $downUrl];
    echo json_encode($res);
    exit();
}

// 返回上传状态
$res = ['status' => $status,'downUrl' => $downUrl];
echo json_encode($res);


运行效果

image.png

猜你想看

电影分享NO.012||《酱园弄·悬案》詹周氏的刀,是反抗还是绝望?
招聘帖里常说的“A/B TesT”到底是个啥?
中国10大名塔你都看过几座?中华文化瑰宝,有生之年一定要去看看
宇宙中的星球为何都是圆的?是什么力量导致的?
今年樱桃产销两端情况如何?“樱桃自由”能够实现吗?
晕车、晕船、晕飞机!难受!外出游玩都成难题!如何预防与治疗?
早上起床后有这5个表现,说明身体很健康!
我的抗衰秘籍,零成本养出少女感
梦幻西游:探讨游戏中装备价值的多元化
个人医保账户究竟有哪些作用?如何用个人账户在药店买药呢?
&quot;造星计划&quot;月考成绩单发布,平均MAU增长率突破400%
油条这样做才好吃,蓬松绵软,口感细腻,吃一次就爱上
很多司机自动挡下长坡不会开,喵哥实车教你操作,防止刹车失灵
演唱会突发意外,张学友回应
医保改革六问:为何要减少个人账户划入?短期吃亏能否换来长远实惠?
轻松摆脱水桶腰!5个瑜伽动作,让你变身水蛇腰美女
毕业旅行成暑期游热门
不论男女,45岁后若有5个特征,偷着乐吧,可能有长寿“潜质”
特斯拉突然涨价,汽车行业要“变天”了?
引力红移是什么?很简单,宇宙大爆炸后的膨胀速率是靠红移测量的

推荐站点