PHP如何上傳文件和下載,你學會了嗎?

第1章 文件上傳

1.1 客戶端上傳設置

​ 在 B/S 程序中文件上傳已經成爲一個常用功能。其目的是客戶可以通過瀏覽器(Browser)將文件上傳到服務器(Server)上的指定目錄。

​ 網絡上常見的支持文件上傳的網站:

​ 各種網盤

​ 頭像

​ 網絡相冊

​ 實名認證

​ 郵件附件

​ 簡單來說就是,Web 開發需要用戶傳遞給服務器文件的都屬於 PHP 的上傳範疇。而服務器端只能接受的份,除非不做這個功能。就像 10086 客服,你只要打過去,它就要接受,不接受只能是服務器繁忙。

PHP中文件上傳的基礎知識:

1)客戶端 form 表單設置

2) 服務器對上傳文件進行操作處理

必須設置的 form 表單項:

<html>
    <head><title>文件上傳</title></head>
    <body>
        <form action="./upload.php"  method="post" enctype="multipart/form-data"><!--必須要寫-->
            <!--typle寫file類型,name必須要寫,名字隨便-->
            選擇文件:<input type="file" name="myfile">
            <input type="submit" value="上傳文件">
        </form>
    </body>
</html>

注意幾個特徵屬性:

1. 一定是 post 方式上傳文件,不可用 get 方式。

2. form 表單中一定要寫enctype="multipart/form-data"。

3. input表單一定要寫name名。

1.2 在服務器端通過PHP處理上傳

上傳文件的接收和處理是通過PHP腳本來處理的,具體需要通過以下三個方面信息:

1)設置 PH 配置文件中的指令:用於精細地調節 PHP 的文件上傳功能。

2)$FILES 多維數組:用於存儲各種與上傳文件有關的信息,其他數據還是使用 $_POST 獲取。

3)PHP 的文件上傳處理函數:用於上傳文件的後續處理。

1)PHP 配置文件中與文件上傳有關的選項。

指令名 默認值 功能描述
file_uploads ON 是否開啓文件上傳
upload_max_filesize 2M 限制PHP處理上傳文件大小的最大值,此值必須小於post_max_size
post_max_size 8M 限制通過POST方法可以接受信息的最大值,也就是整個POST請求的提交值。此值必須大於upload_max_filesize
upload_tmp_dir NULL 上傳文件存放的臨時路徑,可以是絕對路徑。默認NULL則使用系統的臨時目錄。
max_file_uploads 20 文件允許同時上傳的個數

2)$_FILES 多維數組。

超級全局數組 $_FILES

    1、$_FILES["myfile"]["name"]中的值是:客戶端文件系統的文件的名稱。 

    2、$FILES["myfile"]["type"]中的值是:客戶端傳遞的文件的類型。

    3、$_FILES["myfile"]["size"]中的值是:文件的字節的大小。

    4、$_FILES["myfile"]["tmp_name"]中的值是:文件被上傳後在服務器存儲的臨時全路徑。 

    5、$_FILES["myfile"]["error"]中的值是:文件上傳的錯誤代碼-php 4.2以後增加的功能。

關於 error 文件上傳的錯誤代碼:

UPLOAD_ERR_OK
其值爲 0,沒有錯誤發生,文件上傳成功。

UPLOAD_ERR_INI_SIZE
其值爲 1,上傳的文件超過了 php.ini 中 upload_max_filesize 選項限制的值。

UPLOAD_ERR_FORM_SIZE
其值爲 2,上傳文件的大小超過了 HTML 表單中 MAX_FILE_SIZE 選項指定的值。

UPLOAD_ERR_PARTIAL
其值爲 3,文件只有部分被上傳。

UPLOAD_ERR_NO_FILE
其值爲 4,沒有文件被上傳。

UPLOAD_ERR_NO_TMP_DIR
其值爲 6,找不到臨時文件夾。PHP 4.3.10 和 PHP 5.0.3 引進。

UPLOAD_ERR_CANT_WRITE
其值爲 7,文件寫入失敗。PHP 5.1.0 引進。

注意: 以上值在 PHP 4.3.0 之後變成了 PHP 常量。

常見數據格式(MIME)

文件類型 MIME類型
圖片文件 image/gif,image/jpg,image/jpeg,image/png,image/x-png
純文本和HTML text/txt,text/plain,text/html
二進制文件 application/octet-stream
音頻格式 audio/basic
視頻格式 video/mpeg

3) PHP 的文件上傳處理函數

上傳成功的文件會被放置到服務器端臨時目錄下,文件名是隨機生成的臨時文件名。

注:該文件在程序執行完後將自動被刪除掉。在刪除前可以像本地文件一樣操作。

文件上傳處理函數:

is_uploaded_file — 判斷文件是否是通過 HTTP POST 上傳的。

​ 格式:bool is_uploaded_file ( string $filename )

move_uploaded_file — 將上傳的文件移動到新位置。

​ 格式:bool move_uploaded_file ( string $filename , string $destination )

​ 注意:如果目標文件已經存在,將會被覆蓋。

​ 參數說明:文件臨時目錄,要移動到的位置目錄

案例:

1) 設置前端上傳界面

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Document</title>
</head>
<body>
    <form action="doup.php" method="post" enctype="multipart/form-data">
        <input type="file" name="pic">
        <input type="submit" value="上傳">
    </form>
</body>
</html>

2) doup.php 處理上傳到臨時目錄的文件

    //專業搬運工具
    //move_uploaded_file()
    //參數1: 文件臨時目錄  參數2: 要移動到的位置

    //is_uploaded_file() 判斷文件是否是http post提交過來的
    //參數1: 文件臨時目錄
    


    //1.我們保存的路徑按照時間來創建
    //var_dump($_GET);

    //1.1保存的路徑
    $dir='./biran/'.date('Y/m/d/');
    //echo $dir;exit;

    //1.2 判斷文件上傳的路徑是否存在 如果不存在就創建
    if(!file_exists($dir)){
        mkdir($dir,777,true);
    }

    //2.要有個好的文件名 唯一的文件名
    //2.1獲取文件的後綴名
    //2.jpg  jpg 
    $suffix = pathinfo($_FILES['pic']['name'],PATHINFO_EXTENSION);
    //echo $suffix;

    //2.2重新起名
    $filename = date('Ymd').uniqid().mt_rand(0,9999).'.'.$suffix;
    //echo $filename;

    //開始搬運
    //判斷是否是http post 傳遞的文件
    if(!is_uploaded_file($_FILES['pic']['tmp_name'])){
        //不是http post上傳文件
        echo '別整沒用的!!';exit;
    }

    //開始真正的搬運
    if(move_uploaded_file($_FILES['pic']['tmp_name'],$dir.$filename)){
        echo '11111111111';
    }else{
        echo '22222222222';
    }

封裝成函數:

思路:

    function upload(){
        //1.判斷文件上傳錯誤

        //2.判斷你文件上傳的類型是否是你想要的類型
        
        //3.起名字
        
        //4.判斷保存路徑是否存在
        
        //5.判斷是否是http post方式上傳
        
        //6.移動圖片
        
        //7.返回移動成功的圖片名
    }

開始封裝函數:新建 function.php

<?php
    /*
        文件上傳函數
        @param  string  $name  文件上傳文件域的name值
        @param  string  $dir   文件保存路徑
        @param  array   $allow 文件允許上傳的類型
        return  string  $filename 文件名  如果失敗 返回false
     */

    function upload($name,$dir='./upload/',$allow=array('jpg','gif','jpeg','png')){
        //echo $name;exit;
        //var_dump($_FILES);exit;
        //1.判斷文件上傳錯誤
        if($_FILES[$name]['error']>0){
            //echo '上傳錯誤';
            switch($_FILES[$name]['error']){
                case 1:
                    echo '上傳的文件超過了 php.ini 中upload_max_filesize 選項限制的值.';
                    break;
                case 2:
                    echo '上傳文件的大小超過了 HTML 表單中 MAX_FILE_SIZE 選項指定的值';
                    break;
                case 3:
                    echo '文件只有部分被上傳.';
                    break;
                case 4:
                    echo '沒有文件被上傳.';
                    break;
                case 6:
                    echo '找不到臨時文件夾.';
                    break;
                case 7:
                    echo '文件寫入失敗.';
                    break;
            }
            return false;
        }

        //2.判斷你文件上傳的類型是否是你想要的類型
        //2.1允許上傳的類型
        
        //2.2 獲取後綴名
        $suffix = pathinfo($_FILES[$name]['name'],PATHINFO_EXTENSION);
        //echo $suffix;exit;
        //2.3 判斷是否是我們允許上傳的類型
        //var_dump(in_array($suffix,$allow));exit;
        if(!in_array($suffix,$allow)){
            //不允許上傳的類型
            echo  '大哥你的上傳類型不符合';
            return false;
        }
        //3.起名字
        $filename = date('Ymd').uniqid().mt_rand(0,9999).'.'.$suffix;
        //echo $filename;exit;
        
        //4.判斷保存路徑是否存在
        //4.1 得到保存路徑
        
        //4.2 處理保存路徑和後面的斜槓
        $save_path = rtrim($dir,'/');
        $save_path .='/';
    
        //4.3 保存路徑中的時間文件夾處理
        $save_path .=date('Y/m/d/');
    
        //4.4 判斷保存的路徑是否存在
        if(!file_exists($save_path)){
            mkdir($save_path,777,true);
        }
    
        //4.5 拼接一個完整的保存路徑
        $path = $save_path.$filename;
        //echo $path;exit;

        
        //5.判斷是否是httppost方式上傳
        if(!is_uploaded_file($_FILES[$name]['tmp_name'])){
            echo '滾蛋!';
            return false;
        }
        
        //6.移動圖片
        if(!move_uploaded_file($_FILES[$name]['tmp_name'],$path)){
            echo '移動失敗';
            return false;
        }
        
        //7.返回移動成功的圖片名
        return $filename;

    }

調用函數開始上傳:

<?php
    include './function.php';
    //var_dump($_FILES);exit;

    echo upload('file','./leiding',array('jpg','png'));

第2章 多文件上傳

2.1 不同name名稱多文件上傳

當需要上傳多個文件的情況,有兩種實現的解決方法:

1) 使用不同的表單元素。

<input type="file" name="file_a">
<input type="file" name="file_b">
<input type="file" name="file_c">

2) 使用數組格式的表單元素。

<input type="file" name="file[]">
<input type="file" name="file[]">
<input type="file" name="file[]">

第3章 文件下載

1) 對於瀏覽器不識別的文件,可以直接利用 a 連接下載。

    <!-- 因爲他們三個瀏覽器不認識這樣的類型 -->
    <a href="./downlist/1.rar">1.rar</a>
    <a href="./downlist/1.exe">1.exe</a>
    <a href="./downlist/1.avi">1.avi</a>

2) 對於瀏覽器不識別的,可以利用 readfile 函數。

    <!-- 瀏覽器認識這樣的類型,就會被解析 -->
    <a href="./action.php?name=1.html">1.html</a>
    <a href="./action.php?name=1.php">1.php</a>
    <a href="./action.php?name=1.txt">1.txt</a>
    <a href="./action.php?name=1.jpg">1.jpg</a>
//接收一下name值.
$name = $_GET['name'];

//實現下載功能
//強制瀏覽器彈出另存爲對話框
header('content-Disposition:attachment;filename="'.$name.'"');

//此時只是下載了一個空文件,需要利用readfile讀一遍所有的內容.便可下載.
$path = './downlist/'.$name;
readfile($path);
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章