PHP 파일 업로드 백엔드 처리를 위한 실용적인 팁과 방법
인용문: 이전 기사에서 언급했듯이 페이지에 숨겨진 메소드를 사용하여 업로드 페이지를 아름답게 만들 수 있습니다. 그런데 이건 성능상 아무 소용이 없는데, 백그라운드 처리에 뭔가 처리 기술이 있는 게 아닐까요? 소위 백그라운드 기술에는 더 빠른 업로드와 더 큰 파일 업로드가 포함되어야 합니다! 그래서 이번 글에서는 백엔드 처리에는 어떤 스킬이 있는지 알아보겠습니다!
비즈니스 시나리오 1. 업로드할 파일을 하나만 선택하므로 즉각적인 검증 작업을 수행할 필요가 없습니다. 그렇다면 최적화가 전혀 없을 수도 있습니다. 왜냐하면 결국 이 파일을 양식에 넣고 함께 제출한 다음 직접 처리하기만 하면 되기 때문입니다!
비즈니스 시나리오 2: 여러 파일을 업로드해야 하며 파일의 내부 내용을 수시로 확인하여 해당 페이지에 표시해야 합니다. 이 경우에는 사용자가 파일 업로드를 선택한 후 파일에 있는 정보를 읽어야 하기 때문에 즉시 파일을 업로드해야 하고, 마지막에 제출할 때도 파일을 한 번 제출해야 합니다. 분명히 여기에는 반복적인 업로드 작업이 있습니다. 하나는 사용자 시간을 소모하고 다른 하나는 서버 대역폭 리소스를 소모합니다. 최적화, 생각할 수 있는 방법도 매우 간단합니다. 처음 파일을 업로드한 후 파일을 서버에 보관할 수 있으며 실제로 양식이 제출되면 업로드된 임시 파일을 읽으면 됩니다. 예, 이것이 우리의 접근 방식입니다!
비즈니스 시나리오 3은 시나리오 2와 유사하며 여러 파일을 업로드해야 하지만 여러 파일을 별도로 업로드할 수도 있습니다. 즉, 처음에 10M, 두 번째에 10M를 업로드하고 총 10번 업로드했을 수 있습니다. 그러면 서버측에서는 일회성 제출이 업로드 크기 제한을 초과했을 것입니다. 매번 업로드도 가능하고, 제출시에는 짧은 문자만 업로드해주시면 됩니다!
아이디어는 정말 간단해서 문제 없을 것 같습니다. 하지만 제 능력이 한계가 있어서 이걸 처리하는데 정말 시간이 많이 걸렸던 것 같아요! 아래 참고용 샘플 코드를 알려드리겠습니다.
파일 업로드 스킬(단일 업로드된 파일은 서버 측 임시 파일로 존재) 샘플 코드:
1. 페이지 js 처리
//点击选择完成文件后,触发上传文件操作,将文件上传至服务器临时目录 $('.upload-real-file').off().on('change', function(){ if(!$(this).val()){ return false; } var responseObjId = $(this).attr('response-id'); var responseObj = $('#' + responseObjId); $('#Form').ajaxSubmit({ url:'/aa/bb/uploadTmpApkTool', resetForm: false, dataType: 'json', beforeSubmit: function(option){ window.loading = layer.load(2); }, success: function(data, statusText){ layer.close(window.loading); if(data.status == 1){ responseObj.html(data.apkInfoHtml); var parentContainer = responseObj.parent().parent(), nameContainer = parentContainer.find('.file-name-container'); nameContainer.html(data.apkName); nameContainer.attr('title', data.apkName); responseObj.find('.file-tmp').html(data.fileInfo); //将文件信息存放于隐藏域中,以便在提交时能找到 $(submitId).removeAttr('disabled'); }else{ layer.alert(data.info); } }, error: function(data){ layer.close(window.loading); layer.alert('未知错误,请稍后再试!'); } }); return false;//防止dialog 自动关闭 });
2. 당연히 페이지는 파일 정보, 백그라운드 처리 코드(PHP)를 얻어야 합니다.
$apkConfig = $this->_getApkConfig(); $params = $this->getFilteredParam('get'); $subFile = $_FILES['apkToolFiles']; $apkName = $apkInfoHtml = ""; if(empty($subFile)) { $this->ajaxReturn(array('status' => -4, 'info' => '请选择要上传的文件')); } foreach ($subFile['name'] as $subKey => $subVal) { if ($subFile['name'][$subKey]) { $fileData = $this->_getFileData($subFile, $subKey); $checkData = array( 'maxSize' => $apkConfig['FILE_MAX_SIZE'], 'savePath' => $apkConfig['TMP_CHILD_PATH'], 'extArr' => array('apk'), 'releaseName' => str_replace('.apk', '', $fileData['fileName']), //特有 ); $checkResult = $this->_checkFileData($fileData, $checkData); if ($checkResult['status'] != 1) { $this->ajaxReturn($checkResult); } //移动文件 $filePath = $checkData['savePath'] . '/' . $fileData['fileName'] . '.tmp' . genRandStr(6);; $this->_moveUploadedFile($fileData['tmpName'], $filePath); $apkInfo = $this->_apkParser($filePath); //解析 if($apkInfo['UMENG_CHANNEL'] != 'UMENG_CHANNEL_VALUE') { @unlink($filePath); //删除无效文件 $this->ajaxReturn(array('status' => 0, 'info' => "UMENG_CHANNEL的值要是 UMENG_CHANNEL_VALUE才行")); } $tmpFileArr['file_info'] = array( 'name' => $subFile['name'][$subKey], 'type' => $subFile['type'][$subKey], 'tmp_name' => str_replace($apkConfig['TMP_CHILD_PATH'] . '/', '', $filePath), 'error' => $subFile['error'][$subKey], 'size' => $subFile['size'][$subKey], ); //转存该值,不再重复上传文件 } else { $this->ajaxReturn(array('status' => 0, 'info' => "文件不能为空")); } foreach ($apkInfo as $key => $val) { $apkInfoHtml .= "{$key}:{$val} \r\n"; } $apkName = $fileData['fileName']; $version = $apkInfo['versionName']; } $fileInfo = htmlspecialchars(json_encode($tmpFileArr['file_info'])); $fileInfoHtml = "<input name=\"apkToolFileTmp[]\" value='{$fileInfo}' type=\"hidden\"/>"; //一定要输出前使用htmlspecialchars, 否则不能正确显示页面值和获取至正确的文件信息 $this->ajaxReturn(array('status' => 1, 'info' => "上传成功", 'version' => $version, 'item' => $item, 'apkName' => $apkName, 'apkInfoHtml' => $apkInfoHtml, 'fileInfo' => $fileInfoHtml)); }
3. 두 코드 부분의 협력을 통해 이미 마지막에 양식 제출 시 파일을 서버에 제출하지 않으셔도 되고, 서버측에서 처리하실 때에는 이전에 업로드한 임시 파일만 옮기시면 끝입니다!
$('.upload-file-real').attr('disabled', 'disabled'); //提交表单前,禁用上传文件
4. 후속 작업
임시 파일을 서버에 업로드한 후 사용자가 현재 작업을 취소했는지 여부를 판단할 방법이 없습니다. 는 항상 서버에 존재하므로 임시 디렉토리를 정기적으로 정리하는 스크립트가 필요합니다. 물론 이것은 매우 간단합니다. 예를 들어, 하루보다 오래된 파일은 삭제됩니다. 청소 빈도 조절에만 주의하세요!
5. 여담
로그는 정말 중요합니다. 문제가 발생한 곳, 파일이 삭제된 곳, 데이터베이스가 정리된 곳을 잘 기록해야 합니다. 이유, 도움을 청하려면 어디로 가야합니까!
서버의 임시 디렉터리에 파일을 업로드합니다. 백엔드 처리 원리는 매우 간단해 보이지만, 적어도 이 작은 기능을 수행할 때는 주의 깊게 디버깅해야 합니다. 그것을 알아내기 위한 노력!
읽어주셔서 감사합니다. 도움이 되기를 바랍니다. 이 사이트를 지원해 주셔서 감사합니다!
PHP 파일 업로드 백엔드 처리에 대한 실용적인 팁과 방법에 관한 더 많은 기사를 보려면 PHP 중국어 웹사이트를 주목하세요!