>  기사  >  백엔드 개발  >  프론트엔드는 파일의 중단점 재개 업로드를 구현합니다(프런트엔드 파일 제출 + 백엔드 PHP 파일 수신).

프론트엔드는 파일의 중단점 재개 업로드를 구현합니다(프런트엔드 파일 제출 + 백엔드 PHP 파일 수신).

高洛峰
高洛峰원래의
2016-12-26 14:01:251817검색

브레이크포인트 이력서 업로드에 대해 오랫동안 들어왔는데, 프론트엔드에서도 구현할 수 있습니다. 프런트 엔드에서 중단점 이력서 다운로드 구현은 주로 HTML5의 새로운 기능에 의존하므로 일반적으로 이전 브라우저에 대한 지원은 높지 않습니다.

이 기사에서는 일반적인 구현 프로세스를 이해하기 위해 중단점 재개(프런트엔드 파일 제출 + 백엔드 PHP 파일 수신)의 간단한 예를 사용합니다.

먼저 그림을 예로 들어 보겠습니다. 최종 살펴보기

프론트엔드는 파일의 중단점 재개 업로드를 구현합니다(프런트엔드 파일 제출 + 백엔드 PHP 파일 수신).

프론트엔드는 파일의 중단점 재개 업로드를 구현합니다(프런트엔드 파일 제출 + 백엔드 PHP 파일 수신).

1. 약간의 지식 준비

부터 중단점에서 전송을 재개합니다. 중단이 있으면 파일이 있어야 합니다. 분할 프로세스는 하나씩 전달됩니다.

과거에는 파일을 분할할 수 없었으나 HTML5의 새로운 기능이 도입되면서 일반 문자열과 배열의 분할과 유사하게 슬라이스 방식을 사용하여 파일을 분할할 수 있게 되었습니다.

따라서 중단점 재개의 가장 기본적인 구현은 프런트 엔드가 FileList 객체를 통해 해당 파일을 획득하고 지정된 분할 방법에 따라 대용량 파일을 세그먼트로 나눈 다음 이를 백엔드 조각에 전달하는 것입니다. 그런 다음 파일을 하나씩 순서대로 연결합니다.

FileList 객체를 수정한 후 제출해야 합니다. 이전 기사에서는 FileList 객체를 직접 변경할 수 없으므로 직접 전달할 수 없기 때문에 주의해야 할 몇 가지 사항을 배웠습니다. 양식의 .submit() 메소드를 업로드하고 제출하려면 FormData 객체를 결합하여 새 데이터를 생성하고 Ajax를 통해 업로드 작업을 수행해야 합니다.

2. 구현 프로세스

이 예제는 중단점에서 파일 업로드를 재개하는 기본 기능을 구현합니다. 그러나 수동으로 "업로드 일시 중지" 작업이 성공하지 못했습니다. 업로드 중단을 시뮬레이션하고 "업로드 재개"를 경험해 보세요.

그 외 작은 버그가 있을 수 있지만 기본 논리는 거의 동일합니다.

1. 프론트엔드 구현

먼저 파일을 선택하고 선택한 파일 목록 정보를 나열한 다음 업로드 작업을 사용자 정의합니다

(1) 따라서 페이지를 설정합니다. 첫 번째 DOM 구조

<!-- 上传的表单 -->
<form method="post" id="myForm" action="/fileTest.php" enctype="multipart/form-data">
<input type="file" id="myFile" multiple>
<!-- 上传的文件列表 -->
<table id="upload-list">
<thead>
<tr>
<th width="35%">文件名</th>
<th width="15%">文件类型</th>
<th width="15%">文件大小</th>
<th width="20%">上传进度</th>
<th width="15%">
<input type="button" id="upload-all-btn" value="全部上传">
</th>
</tr>
</thead>
<tbody>
</tbody>
</table>
</form>
<!-- 上传文件列表中每个文件的信息模版 -->
<script type="text/template" id="file-upload-tpl"> <tr> <td>{{fileName}}</td> <td>{{fileType}}</td> <td>{{fileSize}}</td> <td class="upload-progress">{{progress}}</td> <td> <input type="button" class="upload-item-btn" data-name="{{fileName}}" data-size="{{totalSize}}" data-state="default" value="{{uploadVal}}"> </td> </tr> </script>

여기서 CSS 스타일을 버리세요

<style type="text/css">
body {
font-family: Arial; }
form {
margin: 50px auto;
width: 600px; }
input[type="button"] {
cursor: pointer; }
table {
display: none;
margin-top: 15px;
border: 1px solid #ddd;
border-collapse: collapse; }
table th {
color: #666; }
table td,
table th {
padding: 5px;
border: 1px solid #ddd;
text-align: center;
font-size: 14px; }
</style>

(2) 다음은 JS 구현 분석

FileList 객체를 통해 파일에 대한 정보를 얻을 수 있습니다

프론트엔드는 파일의 중단점 재개 업로드를 구현합니다(프런트엔드 파일 제출 + 백엔드 PHP 파일 수신).

크기는 파일 크기, 파일 분할은 이것에 의존해야 합니다

여기서의 크기는 바이트 수이므로 인터페이스에서 파일 크기를 표시할 때 다음과 같이 변환할 수 있습니다

// 计算文件大小 size = file.size > 1024 ? file.size / 1024 > 1024 ? file.size / (1024 * 1024) > 1024 ? (file.size / (1024 * 1024 * 1024)).toFixed(2) + &#39;GB&#39; : (file.size / (1024 * 1024)).toFixed(2) + &#39;MB&#39; : (file.size / 1024).toFixed(2) + &#39;KB&#39; : (file.size).toFixed(2) + &#39;B&#39;;

파일 선택 후 파일 정보를 표시하고 템플릿의 데이터를 교체합니다

// 更新文件信息列表 uploadItem.push(uploadItemTpl
.replace(/{{fileName}}/g, file.name)
.replace(&#39;{{fileType}}&#39;, file.type || file.name.match(/\.\w+$/) + &#39;文件&#39;)
.replace(&#39;{{fileSize}}&#39;, size)
.replace(&#39;{{progress}}&#39;, progress)
.replace(&#39;{{totalSize}}&#39;, file.size)
.replace(&#39;{{uploadVal}}&#39;, uploadVal)
);

단, 파일 정보, 이 파일은 이전에 업로드되었을 수 있습니다. 중단점에서 업로드를 재개하려면 인터페이스에서 판단하고 메시지를 표시해야 합니다.

로컬 영역에서 해당 데이터가 있는지 확인하세요. (여기서의 접근 방식은 로컬 기록이 100% 업로드 되었을 때 계속 업로드하지 않고 직접 재업로드하는 방식입니다.)

// 初始通过本地记录,判断该文件是否曾经上传过 percent = window.localStorage.getItem(file.name + &#39;_p&#39;); if (percent && percent !== &#39;100.0&#39;) {
progress = &#39;已上传 &#39; + percent + &#39;%&#39;;
uploadVal = &#39;继续上传&#39;;
}

파일 정보 목록 표시

프론트엔드는 파일의 중단점 재개 업로드를 구현합니다(프런트엔드 파일 제출 + 백엔드 PHP 파일 수신).

클릭하면 업로드가 시작되고 해당 파일을 업로드할 수 있습니다

프론트엔드는 파일의 중단점 재개 업로드를 구현합니다(프런트엔드 파일 제출 + 백엔드 PHP 파일 수신).

파일 업로드 시 파일을 여러 세그먼트로 분할해야 합니다.

예를 들어 여기에 구성된 각 세그먼트는 1024B, 전체 청크 세그먼트(마지막 세그먼트인지 확인하는 데 사용), 청크 세그먼트, 현재 업로드된 비율 등입니다.
꼭 언급해야 할 것은 업로드를 일시정지하는 작업입니다. 사실 아직 구현하지 않았는데 일시정지할 수는 없지만…

프론트엔드는 파일의 중단점 재개 업로드를 구현합니다(프런트엔드 파일 제출 + 백엔드 PHP 파일 수신).

다음 단계는 세분화 과정프론트엔드는 파일의 중단점 재개 업로드를 구현합니다(프런트엔드 파일 제출 + 백엔드 PHP 파일 수신).

// 设置分片的开始结尾 var blobFrom = chunk * eachSize, // 分段开始 blobTo = (chunk + 1) * eachSize > totalSize ? totalSize : (chunk + 1) * eachSize, // 分段结尾 percent = (100 * blobTo / totalSize).toFixed(1), // 已上传的百分比 timeout = 5000, // 超时时间 fd = new FormData($(&#39;#myForm&#39;)[0]);
 
fd.append(&#39;theFile&#39;, findTheFile(fileName).slice(blobFrom, blobTo)); // 分好段的文件 fd.append(&#39;fileName&#39;, fileName); // 文件名 fd.append(&#39;totalSize&#39;, totalSize); // 文件总大小 fd.append(&#39;isLastChunk&#39;, isLastChunk); // 是否为末段 fd.append(&#39;isFirstUpload&#39;, times === &#39;first&#39; ? 1 : 0); // 是否是第一段(第一次上传)
// 上传之前查询是否以及上传过分片 chunk = window.localStorage.getItem(fileName + &#39;_chunk&#39;) || 0;
chunk = parseInt(chunk, 10);

   

文件应该支持覆盖上传,所以如果文件以及上传完了,现在再上传,应该重置数据以支持覆盖(不然后端就直接追加 blob数据了)

// 如果第一次上传就为末分片,即文件已经上传完成,则重新覆盖上传 if (times === &#39;first&#39; && isLastChunk === 1) { window.localStorage.setItem(fileName + &#39;_chunk&#39;, 0);
chunk = 0;
isLastChunk = 0;
}

   

这个 times 其实就是个参数,因为要在上一分段传完之后再传下一分段,所以这里的做法是在回调中继续调用这个上传操作

프론트엔드는 파일의 중단점 재개 업로드를 구현합니다(프런트엔드 파일 제출 + 백엔드 PHP 파일 수신).

接下来就是真正的文件上传操作了,用Ajax上传,因为用到了FormData对象,所以不要忘了在$.ajax({}加上这个配置processData: false

上传了一个分段,通过返回的结果判断是否上传完毕,是否继续上传

success: function(rs) {
rs = JSON.parse(rs); // 上传成功 if (rs.status === 200) { // 记录已经上传的百分比 window.localStorage.setItem(fileName + &#39;_p&#39;, percent); // 已经上传完毕 if (chunk === (chunks - 1)) {
$progress.text(msg[&#39;done&#39;]);
$this.val(&#39;已经上传&#39;).prop(&#39;disabled&#39;, true).css(&#39;cursor&#39;, &#39;not-allowed&#39;); if (!$(&#39;#upload-list&#39;).find(&#39;.upload-item-btn:not(:disabled)&#39;).length) {
$(&#39;#upload-all-btn&#39;).val(&#39;已经上传&#39;).prop(&#39;disabled&#39;, true).css(&#39;cursor&#39;, &#39;not-allowed&#39;);
}
} else { // 记录已经上传的分片 window.localStorage.setItem(fileName + &#39;_chunk&#39;, ++chunk);
$progress.text(msg[&#39;in&#39;] + percent + &#39;%&#39;); // 这样设置可以暂停,但点击后动态的设置就暂停不了.. // if (chunk == 10) { // isPaused = 1; // } console.log(isPaused);
if (!isPaused) {
startUpload();
}
}
}
// 上传失败,上传失败分很多种情况,具体按实际来设置 else if (rs.status === 500) {
$progress.text(msg[&#39;failed&#39;]);
}
},
error: function() {
$progress.text(msg[&#39;failed&#39;]);
}

   

2. 后端实现

要注意一下,通过FormData对象上传的文件对象,在PHP中也是通过$_FILES全局对象获取的,还有为了避免上传后文件中文的乱码,用一下iconv
断点续传支持文件的覆盖,所以如果已经存在完整的文件,就将其删除

// 如果第一次上传的时候,该文件已经存在,则删除文件重新上传 if ($isFirstUpload == &#39;1&#39; && file_exists(&#39;upload/&#39;. $fileName) && filesize(&#39;upload/&#39;. $fileName) == $totalSize) {
unlink(&#39;upload/&#39;. $fileName);
}

   

使用上述的两个方法,进行文件信息的追加,别忘了加上 FILE_APPEND 这个参数~

// 继续追加文件数据 if (!file_put_contents(&#39;upload/&#39;. $fileName, file_get_contents($_FILES[&#39;theFile&#39;][&#39;tmp_name&#39;]), FILE_APPEND)) {
$status = 501;
} else { // 在上传的最后片段时,检测文件是否完整(大小是否一致) if ($isLastChunk === &#39;1&#39;) { if (filesize(&#39;upload/&#39;. $fileName) == $totalSize) {
$status = 200;
} else {
$status = 502;
}
} else {
$status = 200;
}
}

   

一般在传完后都需要进行文件的校验吧,所以这里简单校验了文件大小是否一致。

根据实际需求的不同有不同的错误处理方法,这里就先不多处理了

完整的PHP部分

 0) {
$status = 500;
} else { // 此处为一般的文件上传操作 // if (!move_uploaded_file($_FILES['theFile']['tmp_name'], 'upload/'. $_FILES['theFile']['name'])) { // $status = 501; // } else { // $status = 200; // } // 以下部分为文件断点续传操作 // 如果第一次上传的时候,该文件已经存在,则删除文件重新上传 if ($isFirstUpload == &#39;1&#39; && file_exists(&#39;upload/&#39;. $fileName) && filesize(&#39;upload/&#39;. $fileName) == $totalSize) {
unlink(&#39;upload/&#39;. $fileName);
} // 否则继续追加文件数据 if (!file_put_contents('upload/'. $fileName, file_get_contents($_FILES['theFile']['tmp_name']), FILE_APPEND)) {
$status = 501;
} else { // 在上传的最后片段时,检测文件是否完整(大小是否一致) if ($isLastChunk === '1') { if (filesize('upload/'. $fileName) == $totalSize) { $status = 200;
} else {
$status = 502;
}
} else {
$status = 200;
}
}
} echo json_encode(array( 'status' => $status, 'totalSize' => filesize('upload/'. $fileName), 'isLastChunk' => $isLastChunk
)); ?>

   

更多프론트엔드는 파일의 중단점 재개 업로드를 구현합니다(프런트엔드 파일 제출 + 백엔드 PHP 파일 수신).相关文章请关注PHP中文网!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.