天天看點

檔案上傳小技巧/後端處理【以php示例】

  引語:在上一篇文章中說到,在頁面中可以用隐藏的方式讓你的上傳頁面看起來漂亮。但是這對于性能來說,并沒有什麼卵用,那麼在背景的進行中,難道就沒有一些處理技巧麼?所謂背景的技巧,應該要包括上傳得快一點,上傳的檔案大一點!那麼,本文就來說說,後端處理都有些什麼技巧吧!

  業務場景一、我們隻會選擇一個單個的檔案上傳,而且不需要做一些即時的驗證工作。那麼,也許并沒有什麼優化可言了,因為,最後你要做的,隻是将這個檔案放在表單裡最後一起送出,直接處理即可!

  業務場景二、需要上傳多個檔案,而且需要時時驗證檔案内部内容,并時行相應頁面顯示。對于這種況,在使用者選擇了上傳檔案之後,我們需要立即将檔案上傳,因為我們需要讀取檔案裡的資訊,在最後送出的時候,我們也需要送出一次檔案。很明顯,在這裡是存在一個重複上傳的工作的,一個耗費使用者時間,二個是耗費伺服器帶寬資源!優化,能夠想得到的方法也很簡單,能不能在第一次上傳完檔案之後,就将檔案保留在伺服器,真正送出表單的時候,去讀取這個已經被上傳的臨時檔案即可。是的,這就是我們的處理思路!

  業務場景三、與場景二類似,需要上傳多個檔案,但是多個檔案可能是分開上傳的。即我們可能第一次上傳了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. 題外話

    日志真的很重要,哪裡出錯了,哪裡删除檔案了,哪裡清理資料庫了,一定要做好記錄,否則,到時查找原因時,到哪裡去喊救命!

    注意臨時檔案的唯一性,不要形成覆寫哦,這樣檔案直接出錯可不是好事,使用随機數也也隻是減小了出現這情況的機率,但其實也有這種可能出現!(我是在等待出現這個問題了再去解決,感覺是個坑)

  

  上傳檔案到伺服器臨時目錄,後端處理原理看起來很簡單,但是也需要你仔細調試哦!

不要害怕今日的苦,你要相信明天,更苦!

繼續閱讀