webuploader实现大文件上传

本文介绍了一个使用Webuploader插件的大文件(视频、音频)上传方案,详细解释了如何配置插件参数,实现文件切片上传,并展示了如何在PHP后端进行分片合并,确保上传过程的稳定性和效率。

目前在公司项目里遇到了需要上传大文件(视频、音频)的情况,特此记录。

此次项目引用了一款名为Webuploader的插件。官网:https://dwz.cn/fgXtRtnu

html代码:

<html>
<head>
    <meta charset="utf-8">
    <script src="jquery-2.1.1.js"></script>
    <link rel="stylesheet" type="text/css" href="webuploader.css">
    <script type="text/javascript" src="webuploader.js"></script>
</head>
<body>
 
<style>
.progress-bar{background-color:red}
</style>
 
<div id="uploader" class="wu-example">    
    <div id="thelist" class="uploader-list"></div>
    <div class="btns">
        <div id="picker">选择文件</div>
        <button id="ctlBtn" class="btn btn-default">开始上传</button>
    </div>
</div>
 
<script>
$(function(){
    var uploader = WebUploader.create({
        
        swf: 'Uploader.swf',
        
        server: 'http://www.test.loc/webuploader/upload.php',
        
        pick: '#picker',
        
        resize: false,
        
        chunked: true           //是否切片
    });
    var $list = $('#thelist'),
        state = 'pending';
    
    uploader.on( 'fileQueued', function( file ) {
        $list.append( '<div id="' + file.id + '" class="item">' +
            '<h4 class="info">' + file.name + '</h4>' +
            '<p class="state">已添加...</p>' +
        '</div>' );
    });
 
    
    uploader.on( 'uploadProgress', function( file, percentage ) {
        var $li = $( '#'+file.id ),
            $percent = $li.find('.progress .progress-bar');
 
        
        if ( !$percent.length ) {
            $percent = $('<div class="progress progress-striped active">' +
              '<div class="progress-bar" role="progressbar" style="width: 0%;height:30px;">' +
              '</div>' +
            '</div>').appendTo( $li ).find('.progress-bar');
        }
 
        $li.find('p.state').text('上传中');
 
        $percent.css( 'width', percentage * 100 + '%' );
    });
 
    uploader.on( 'uploadSuccess', function( file ) {
        $( '#'+file.id ).find('p.state').text('已上传');
    });
 
    uploader.on( 'uploadError', function( file ) {
        $( '#'+file.id ).find('p.state').text('上传出错');
    });
 
    uploader.on( 'uploadComplete', function( file ) {
        $( '#'+file.id ).find('.progress').fadeOut();
    });
    
    $('#ctlBtn').on( 'click', function() {
        if ( state === 'uploading' ) {
            uploader.stop(true);
            state = "pending";
        } else {
            uploader.upload();
            state = "uploading";
        }
    });
})
</script>
 
</body>
</html>

 

 

 

后台upload.php

<?php
header('Access-Control-Allow-Origin:*');
header('Access-Control-Allow-Headers: X-Requested-With,X_Requested_With');
 
$aFiles = getUploadFiles();
 
saveMultiFiles($aFiles[0]);
 
function getUploadFiles()
{
    $aFiles      = $_FILES;
    $aMultiFiles = array();
 
    // 判断是否是分片上传
    $iChunk  = isset($_REQUEST["chunk"]) ? intval($_REQUEST["chunk"]) : 0;
    $iChunks = isset($_REQUEST["chunks"]) ? intval($_REQUEST["chunks"]) : 0;
 
    foreach ($aFiles as $sKey => $mFiles) {
        if (is_array($mFiles['name'])) {
            $iCnt = count($mFiles['name']);
 
            for ($i = 0; $i < $iCnt; ++$i) {
                $aMultiFiles[] = array(
                    'key'      => $sKey . '_' . $i,
                    'name'     => $mFiles['name'][$i],
                    'tmp_name' => $mFiles['tmp_name'][$i],
                    'error'    => $mFiles['error'][$i],
                    'size'     => $mFiles['size'][$i],
                    'chunk'    => $iChunk,
                    'chunks'    => $iChunks,
                );
            }
        } else {
            $aMultiFiles[] = array(
                'key'      => $sKey,
                'name'     => $mFiles['name'],
                'tmp_name' => $mFiles['tmp_name'],
                'error'    => $mFiles['error'],
                'size'     => $mFiles['size'],
                'chunk'    => $iChunk,
                'chunks'   => $iChunks,
            );
        }
    }
 
    return $aMultiFiles;
}
 
/**
  * 将临时文件合并成正式文件
  */
function saveMultiFiles($aFile)
{
    $tmp_file_path = './tmp';
 
    $p_sName         = $aFile['name'];
    $p_sNameFilename = pathinfo($p_sName, PATHINFO_FILENAME);
    $p_sFilePath     = $tmp_file_path.DIRECTORY_SEPARATOR.$p_sNameFilename;
 
    $p_sFilenamePath = $tmp_file_path.DIRECTORY_SEPARATOR.$p_sName;
    if (!file_exists($p_sFilenamePath)) {
        fopen($p_sFilenamePath, "w");
    }
 
    $p_sTmpName = $aFile['tmp_name'];
    $p_iError   = $aFile['error'];
    $p_iSize    = $aFile['size'];
    $iChunk     = $aFile['chunk'] ;
    $iChunks    = $aFile['chunks'];
    $iError     = 0;
    
 
    if ($p_iError > 0) {
        // 文件上传出错
        $iError  = 1;
        $mReturn = '文件上传出错';
        break;
    }
 
    if (!is_uploaded_file($p_sTmpName)) {
        $iError  = 2;
        $mReturn = 'upload error, use http post to upload';
        break;
    }
 
    $oFInfo    = finfo_open();
    $sMimeType = finfo_file($oFInfo, $p_sTmpName, FILEINFO_MIME_TYPE);
 
    finfo_close($oFInfo);
    
    $sExtension = pathinfo($p_sName, PATHINFO_EXTENSION);
    
    if (empty($sExtension)) {
        $iError  = 2;
        $mReturn = 'upload error, The file does not have an extension ';
        break;
    }
 
    if (!$in = @fopen($p_sTmpName, "rb")) {
        $iError  = 1;
        $mReturn = "Failed to open input stream.";
        break;
    }
 
    if (!$out = @fopen("{$p_sFilePath}_{$iChunk}.parttmp", "wb")) {
        $iError  = 1;
        $mReturn = "Failed to open output stream.";
        break;
    }
 
    while ($buff = fread($in, 4096)) {
        fwrite($out, $buff);
    }
    @fclose($out);
    @fclose($in);
 
    rename("{$p_sFilePath}_{$iChunk}.parttmp", "{$p_sFilePath}_{$iChunk}.part");
 
    $done  = true;
    for ($index = 0; $index < $iChunks; $index++) {
        if (!file_exists("{$p_sFilePath}_{$index}.part")) {
            $done = false;
            break;
        }
    }
    
 
    if ($done) {
        
        $sDestFile = './upload/'.time().'.'.$sExtension;      //合并文件地址
 
        if (!$out = @fopen($sDestFile, "wb")) {
            $iError  = 1;
            $mReturn = "1Failed to open output stream.";
            break;
        }
        
        $sFileSize = 0;
 
        if (flock($out, LOCK_EX)) {
            for ($index = 0; $index < $iChunks; $index++) {
                if (!$in = @fopen("{$p_sFilePath}_{$index}.part", "rb")) {
                    break;
                }
 
                while ($buff = fread($in, 4096)) {
                    fwrite($out, $buff);
                }
                @fclose($in);
                @unlink("{$p_sFilePath}_{$index}.part");
            }
            flock($out, LOCK_UN);
        }
        @fclose($out);
 
        // 删除临时文件
        @unlink($p_sFilenamePath);
 
    }
 
    return true;
}

 

 

 

06-22
### 得物技术栈及开发者文档分析 得物作为一家专注于潮流商品的电商平台,其技术栈和开发者文档主要围绕电商平台的核心需求展开。以下是对得物技术栈及相关开发资源的详细解析: #### 1. 技术栈概述 得物的技术栈通常会涵盖前端、后端、移动应用开发以及大数据处理等多个领域。以下是可能涉及的主要技术栈[^3]: - **前端开发**: 前端技术栈可能包括现代框架如 React 或 Vue.js,用于构建高效、响应式的用户界面。此外,还会使用 Webpack 等工具进行模块化打包和优化。 - **后端开发**: 后端技术栈可能采用 Java Spring Boot 或 Node.js,以支持高并发和分布式架构。数据库方面,MySQL 和 Redis 是常见的选择,分别用于关系型数据存储和缓存管理。 - **移动应用开发**: 得物的移动应用开发可能基于原生技术(如 Swift/Kotlin)或跨平台框架(如 Flutter)。这有助于确保移动端应用的性能和用户体验一致性。 - **大数据云计算**: 在大数据处理方面,得物可能会使用 Hadoop 或 Spark 进行数据挖掘和分析。同时,依托云服务提供商(如阿里云或腾讯云),实现弹性扩展和资源优化。 #### 2. 开发者文档分析 类似于引用中提到的 Adobe 开发者文档模板[^2],得物也可能提供一套完整的开发者文档体系,以支持内部团队协作和外部开发者接入。以下是开发者文档可能包含的内容: - **API 文档**: 提供 RESTful API 或 GraphQL 的详细说明,帮助开发者快速集成得物的功能模块,例如商品搜索、订单管理等。 - **SDK 集成指南**: 针对不同平台(如 iOS、Android 或 Web)提供 SDK 下载和集成教程,简化第三方应用的开发流程。 - **技术博客**: 分享得物在技术实践中的经验成果,例如如何优化图片加载速度、提升应用性能等。 - **开源项目**: 得物可能将部分技术成果开源,供社区开发者学习和贡献。这不仅有助于提升品牌形象,还能吸引更多优秀人才加入。 #### 3. 示例代码 以下是一个简单的示例代码,展示如何通过 RESTful API 调用得物的商品搜索功能(假设接口已存在): ```python import requests def search_items(keyword, page=1): url = "https://api.dewu.com/v1/items/search" headers = { "Authorization": "Bearer YOUR_ACCESS_TOKEN", "Content-Type": "application/json" } params = { "keyword": keyword, "page": page, "size": 10 } response = requests.get(url, headers=headers, params=params) if response.status_code == 200: return response.json() else: return {"error": "Failed to fetch data"} # 调用示例 result = search_items("Air Jordan", page=1) print(result) ``` 此代码片段展示了如何通过 Python 请求得物的 API,并获取指定关键词的商品列表。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值