JavaScript+PHP实现视频文件分片上传的示例代码

2024-04-19 0 296
目录
  • 摘要
  • 上代码
  • 程序目录

摘要

视频文件分片上传,整体思路是利用JavaScript将文件切片,然后循环调用上传接口 upload.php 将切片上传到服务器。这样将由原来的一个大文件上传变为多个小文件同时上传,节省了上传时间,这就是文件分片上传的其中一个好处。

JavaScript+PHP实现视频文件分片上传的示例代码

上代码

index.html

通过前端将文件对象切分成多个小块,然后依次将这些小块的文件对象上传到服务器。

<!DOCTYPE html>
<html lang=\”en\”>
<head>
<meta charset=\”UTF-8\”>
<meta name=\”viewport\” content=\”width=device-width, initial-scale=1.0\”>
<title>视频文件分片上传</title>
<style>
*{
padding: 0;
margin: 0;
}
.title {
text-align: center;
font-size: 25px;
margin-top: 50px;
}
.video_upload {
width: 500px;
height: 60px;
background: #eee;
margin: 30px auto 0;
border: 2px dashed #ccc;
border-radius: 10px;
position: relative;
cursor: pointer;
text-align: center;
font-size: 25px;
line-height: 60px;
color: #666;
}
#fileInput {
width: 100%;
height: 100%;
position: absolute;
left: 0;
top: 0;
opacity: 0;
cursor: pointer;
}
#uploadButton {
width: 130px;
height: 40px;
border: none;
outline: none;
border-radius: 10px;
font-size: 17px;
margin: 10px auto;
}
#ret {
text-align: center;
font-size: 16px;
margin-top: 20px;
}
#ret video {
width: 450px;
}
</style>
</head>
<body>

<p class=\”title\”>javaScript+PHP实现视频文件分片上传</p>
<div class=\”video_upload\”>
<span class=\”text\”> + </span>
<input type=\”file\” id=\”fileInput\” accept=\”video/*\”>
</div>
<button id=\”uploadButton\” style=\”display:none;\”>开始上传</button>
<p id=\”ret\”></p>

<script>

// 定义全局变量
let videoFile = null;
let chunkSize = 1024 * 1024; // 1MB 分片大小

// 当文件选择框的值改变时触发该函数
function handleFileSelect(event) {
const fileList = event.target.files;
if (fileList.length > 0) {
videoFile = fileList[0];
console.log(\”选择了文件: \”, videoFile.name);
document.querySelector(\’.video_upload .text\’).textContent = videoFile.name;
document.querySelector(\’#uploadButton\’).style.display = \’block\’;
}
}

// 分片并上传文件
async function uploadFile() {
if (!videoFile) {
console.error(\”请选择一个视频文件\”);
return;
}

const fileSize = videoFile.size;
let start = 0;
let end = Math.min(chunkSize, fileSize);
let chunkIndex = 0;

// 获取文件名
const fileName = videoFile.name;

while (start < fileSize) {
const chunk = videoFile.slice(start, end); // 从文件中截取一个分片

// 使用FormData来构建multipart/form-data格式的请求体
const formData = new FormData();
formData.append(\’file\’, chunk);
formData.append(\’chunkIndex\’, chunkIndex);
formData.append(\’fileName\’, fileName); // 将文件名作为 formData 的一部分

try {
const response = await fetch(\’upload.php\’, {
method: \’POST\’,
body: formData
});

if (!response.ok) {
throw new Error(\’上传失败\’);
}

console.log(\’上传分片 \’, chunkIndex, \’ 成功\’);
} catch (error) {
console.error(\’上传分片 \’, chunkIndex, \’ 失败: \’, error.message);
return;
}

start = end;
end = Math.min(start + chunkSize, fileSize);
chunkIndex++;
}

console.log(\’文件上传完成\’);

// 上传完成后发送通知给服务器进行合并
notifyServerForMerge(fileName);
}

// 发送通知给服务器进行合并
async function notifyServerForMerge(fileName) {
try {
const response = await fetch(\’merge_chunks.php\’, {
method: \’POST\’,
headers: {
\’Content-Type\’: \’application/json\’
},
body: JSON.stringify({ fileName: fileName })
});

if (!response.ok) {
throw new Error(\’无法通知服务器进行合并\’);
}

const res_data = await response.json();

console.log(\’已通知服务器进行合并\’);
document.querySelector(\’.video_upload .text\’).textContent = \’分片合并完成!\’;
document.querySelector(\’#ret\’).innerHTML = \'<video autoplay controls src=\”\’+res_data.filePath+\’\”></video>\’;
document.querySelector(\’#uploadButton\’).style.display = \’none\’;
} catch (error) {
console.error(\’通知服务器进行合并时发生错误: \’, error.message);
}
}

// 注册文件选择框的change事件
document.getElementById(\’fileInput\’).addEventListener(\’change\’, handleFileSelect);

// 注册上传按钮的click事件
document.getElementById(\’uploadButton\’).addEventListener(\’click\’, uploadFile);
</script>

</body>
</html>

upload.php

这个是用于接收前端传过来的每一段分片,然后上传到 uploads 文件夹,上传之后就是一段一段的小分片。

<?php

// 设置允许跨域访问
header(\”Access-Control-Allow-Origin: *\”);
header(\”Access-Control-Allow-Methods: POST\”);

// 检查是否接收到文件和分片索引
if (isset($_FILES[\’file\’][\’error\’]) && isset($_POST[\’chunkIndex\’]) && isset($_POST[\’fileName\’])) {

$error = $_FILES[\’file\’][\’error\’];
$chunkIndex = $_POST[\’chunkIndex\’];
$fileName = $_POST[\’fileName\’]; // 获取文件名

// 检查是否有错误
if ($error !== UPLOAD_ERR_OK) {
http_response_code(500);
echo json_encode(array(
\’error\’ => \’文件上传失败\’
));
exit();
}

// 设置存储目录和文件名
$uploadDir = \’./uploads/\’;
$filePath = $uploadDir . $fileName . \’.\’ . $chunkIndex;

// 将分片移动到指定的目录
if (move_uploaded_file($_FILES[\’file\’][\’tmp_name\’], $filePath)) {

echo json_encode(array(
\’success\’ => \’分片上传成功\’
));
} else {

http_response_code(500);
echo json_encode(array(
\’error\’ => \’分片上传失败\’
));
}
} else {

http_response_code(400);
echo json_encode(array(
\’error\’ => \’缺少文件、分片索引或文件名\’
));
}

?>

merge_chunks.php

这个是用来合并分片的,当前端完成上传分片的操作,前端会异步告诉服务器你已经完成所有分片的上传,接下来将每个分片名告诉合并程序完成所有分片的合并,合并之后就是一个完整的视频文件。

<?php

// 设置允许跨域访问
header(\”Access-Control-Allow-Origin: *\”);
header(\”Access-Control-Allow-Methods: POST\”);
header(\”Content-Type: application/json\”);

// 获取请求体中的文件名
$data = json_decode(file_get_contents(\”php://input\”) , true);
$fileName = isset($data[\’fileName\’]) ? $data[\’fileName\’] : null;
if ($fileName) {

$uploadDir = \’./uploads/\’;
$finalFilePath = $uploadDir . $fileName;
$totalChunks = count(glob($uploadDir . $fileName . \’.*\’));

// 检查是否所有分片都已上传
if ($totalChunks > 0) {

// 所有分片都已上传,开始合并
$finalFile = fopen($finalFilePath, \’wb\’);

// 逐个读取分片并写入最终文件
for ($i = 0; $i < $totalChunks; $i++) {
$chunkFilePath = $uploadDir . $fileName . \’.\’ . $i;
$chunkFile = fopen($chunkFilePath, \’rb\’);
stream_copy_to_stream($chunkFile, $finalFile);
fclose($chunkFile);
unlink($chunkFilePath); // 删除已合并的分片

}

fclose($finalFile);
http_response_code(200);
echo json_encode(array(
\’success\’ => \’文件合并成功\’,
\’filePath\’ => $finalFilePath
));
} else {

http_response_code(400);
echo json_encode(array(
\’error\’ => \’没有上传的分片\’
));
}
} else {

http_response_code(400);
echo json_encode(array(
\’error\’ => \’缺少文件名\’
));
}
?>

程序目录

请自行创建 uploads 目录。

JavaScript+PHP实现视频文件分片上传的示例代码

以上就是JavaScript+PHP实现视频文件分片上传的示例代码的详细内容,更多关于JavaScript+PHP视频文件上传的资料请关注悠久资源网其它相关文章!

您可能感兴趣的文章:

  • PHP+JS实现大文件切片上传功能实现实例源码
  • thinkphp5 + ajax 使用formdata提交数据(包括文件上传) 后台返回json完整实例
  • JS+php后台实现文件上传功能详解
  • php+croppic.js实现剪切上传图片功能
  • PHP+JavaScript实现无刷新上传图片

收藏 (0) 打赏

感谢您的支持,我会继续努力的!

打开微信/支付宝扫一扫,即可进行扫码打赏哦,分享从这里开始,精彩与您同在
点赞 (0)

悠久资源 PHP JavaScript+PHP实现视频文件分片上传的示例代码 https://www.u-9.cn/biancheng/php/188363.html

常见问题

相关文章

发表评论
暂无评论
官方客服团队

为您解决烦忧 - 24小时在线 专业服务