成人无码视频,亚洲精品久久久久av无码,午夜精品久久久久久毛片,亚洲 中文字幕 日韩 无码

資訊專欄INFORMATION COLUMN

考慮 PHP 5.0~5.6 各版本兼容性的 cURL 文件上傳

NikoManiac / 3239人閱讀

摘要:最近做的一個需求,要通過調(diào)用,以格式上傳文件。傳統(tǒng)上,的支持通過在數(shù)組數(shù)據(jù)中,使用文件全路徑的語法附加文件,供讀取上傳。的默認值為,的默認值為。對于而言,手動設(shè)置為是毫無意義的。命名空間或以下的版本沒有命名空間。

最近做的一個需求,要通過PHP調(diào)用cURL,以multipart/form-data格式上傳文件。踩坑若干,夠一篇文章了。

重要警告

沒事不要讀PHP的官方中文文檔!版本跟不上坑死你!

不同版本PHP之間cURL的區(qū)別

PHP的cURL支持通過給CURL_POSTFIELDS傳遞關(guān)聯(lián)數(shù)組(而不是字符串)來生成multipart/form-data的POST請求。

傳統(tǒng)上,PHP的cURL支持通過在數(shù)組數(shù)據(jù)中,使用“@+文件全路徑”的語法附加文件,供cURL讀取上傳。這與命令行直接調(diào)用cURL程序的語法是一致的:

curl_setopt(ch, CURLOPT_POSTFIELDS, array(
    "file" => "@".realpath("image.png"), 
)); 
equals
$ curl -F "file=@/absolute/path/to/image.png" 

但PHP從5.5開始引入了新的CURLFile類用來指向文件。CURLFile類也可以詳細定義MIME類型、文件名等可能出現(xiàn)在multipart/form-data數(shù)據(jù)中的附加信息。PHP推薦使用CURLFile替代舊的@語法:

curl_setopt(ch, CURLOPT_POSTFIELDS, [
    "file" => new CURLFile(realpath("image.png")), 
]); 

PHP 5.5另外引入了CURL_SAFE_UPLOAD選項,可以強制PHP的cURL模塊拒絕舊的@語法,僅接受CURLFile式的文件。5.5的默認值為false,5.6的默認值為true。

但是坑的一點在于:@語法在5.5就已經(jīng)被打了deprecated,在5.6中就直接被刪除了(會產(chǎn)生 ErorException: The usage of the @filename API for file uploading is deprecated. Please use the CURLFile class instead)。

對于PHP 5.6+而言,手動設(shè)置CURL_SAFE_UPLOAD為false是毫無意義的。根本不是字面意義理解的“設(shè)置成false,就能開啟舊的unsafe的方式”——舊的方式已經(jīng)作為廢棄語法徹底不存在了。PHP 5.6+ == CURLFile only,不要有任何的幻想。

我的部署環(huán)境是5.4(僅@語法),但開發(fā)環(huán)境是5.6(僅CURLFile)。都沒有壓在5.5這個兩者都支持過渡版本上,結(jié)果就是必須寫出帶有環(huán)境判斷的兩套代碼。

現(xiàn)在問題來了……(挖掘機滾遠點!)

環(huán)境判斷:小心魔法數(shù)字!

我見過這種環(huán)境判斷的代碼:

if (version_compare(phpversion(), "5.4.0") >= 0)

我對這種代碼的評價只有一個字:

這個判斷掉入了典型的魔法數(shù)字陷阱。版本號莫名其妙的出現(xiàn)在代碼之中,不查半天PHP手冊和更新歷史,很難明白作者被卡在了哪個功能的變更上。

代碼應(yīng)該回歸本源。我們的實際需求其實是:有CURLFile就優(yōu)先采用,沒有再退化到傳統(tǒng)@語法。那么代碼就來了:

if (class_exists("CURLFile")) {
    $field = array("fieldname" => new CURLFile(realpath($filepath)));
} else {
    $field = array("fieldname" => "@" . realpath($filepath));
}
建議明確指定的退化選項

從可靠的角度,推薦指定CURL_SAFE_UPLOAD的值,明確告知php是容忍還是禁止舊的@語法。注意在低版本PHP中CURLOPT_SAFE_UPLOAD常量本身可能不存在,需要判斷:

if (class_exists("CURLFile")) {
    curl_setopt($ch, CURLOPT_SAFE_UPLOAD, true);
} else {
    if (defined("CURLOPT_SAFE_UPLOAD")) {
        curl_setopt($ch, CURLOPT_SAFE_UPLOAD, false);
    }
}
cURL選項設(shè)置的順序

不管是curl_setopt()單發(fā)還是curl_setopt_array()批量,cURL的選項總是設(shè)置一個生效一個,而設(shè)置好的選項立刻就會影響cURL在設(shè)置后續(xù)選項時的行為。

例如CURLOPT_SAFE_UPLOAD就和CURLOPT_POSTFIELDS的行為有關(guān)。如果先設(shè)置CURLOPT_POSTFIELDS再設(shè)置CURLOPT_SAFE_UPLOAD,那么后者的約束作用就不會生效。因為設(shè)置前者時cURL就已經(jīng)把數(shù)據(jù)實際的識讀處理完畢了!

cURL有那么幾個選項存在這種坑,務(wù)必小心。還好這種存在“依賴關(guān)系”的選項不多,機制也不復雜,簡單處理即可。我的方法是先批量設(shè)置所有的選項,然后直到curl_exec()的前一刻才用curl_setopt()單發(fā)設(shè)置CURLOPT_POSTFIELDS。

實際上在curl_setopt_array()用的數(shù)組中,保證CURLOPT_POSTFIELDS的位置在后邊也是可靠的。PHP的關(guān)聯(lián)數(shù)組是有順序保障的,我們也可以假設(shè)curl_setopt_array()內(nèi)部的執(zhí)行順序一定是從頭到尾按順序[注A],所以盡可放心。

我的做法只是在代碼表現(xiàn)上加個多余的保險,突出強調(diào)順序的重要性防以后手賤。

命名空間

PHP 5.2或以下的版本沒有命名空間。代碼中用到了空間分隔符就會引發(fā)解析器錯誤。要照顧PHP 5.2其實容易想,放棄命名空間即可。

要注意的反倒是有命名空間的PHP 5.3+。無論是調(diào)用CURLFile還是用class_exists()判斷CURLFile的存在性,都推薦寫成CURLFile明確指定頂層空間,防止代碼包裹在命名空間內(nèi)的時候崩掉。

注解
  

【注A】 好吧我知道assume不是件好事,不過有些實在過分淺顯的事實,就容我下個最低限度的斷言吧

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://m.hztianpu.com/yun/20859.html

相關(guān)文章

  • WordPress主題 Modown 6.2+Erphpdown 11.7虛擬素材資源付費下載

    wordpress主題 modown 6.2+Erphpdown 11.7虛擬素材資源付費下載源碼名稱:WordPress主題 modown 6.2+Erphpdown 11.7虛擬素材資源付費下載兼容版本:全部版本適配編碼:GBK BIG5 UTF8SC UTF8TC演示地址:http://demo.mobantu.com/modown/(官方演示站,以截圖為準)安裝環(huán)境:安裝要求:PHP 5....

    番茄西紅柿 評論0 收藏2637
  • 使用php調(diào)用微信接口上傳永久素材

    摘要:功能需求公司新開的公眾號需要將公司平臺現(xiàn)在的所有精品文章都導入,手動導入會有很多的工作量,所以采用自動化同步文章的方式來達到效果開發(fā)說明微信提供了新增永久素材的接口,本次功能是基于這個接口進行數(shù)據(jù)同步的使用到的接口獲取永久素材列表接口新增永 功能需求 公司新開的公眾號需要將公司平臺現(xiàn)在的所有精品文章都導入,手動導入會有很多的工作量,所以采用自動化同步文章的方式來達到效果 開發(fā)說明 微信...

    niuxiaowei111 評論0 收藏0
  • 阿里云 ECS 部署:nginx+MySQL+Laravel+PHP7+Redis+Node.js

    摘要:修改增加注意中的應(yīng)為你的項目名稱,應(yīng)為你的服務(wù)器公網(wǎng)。配置完之后重啟,使上面的配置項生效。安裝添加下面鏈接中的源,然后安裝番外安裝如果已經(jīng)下了想升級成的朋友可執(zhí)行以下代碼把之前的掉安裝現(xiàn)在的 版本:ubuntu 14.04(64位) 1.安裝 Nginx(version:1.9) 1、首先添加nginx_signing.key wget http://nginx.org/keys/ng...

    dingding199389 評論0 收藏0
  • 阿里云 ECS 部署:nginx+MySQL+Laravel+PHP7+Redis+Node.js

    摘要:修改增加注意中的應(yīng)為你的項目名稱,應(yīng)為你的服務(wù)器公網(wǎng)。配置完之后重啟,使上面的配置項生效。安裝添加下面鏈接中的源,然后安裝番外安裝如果已經(jīng)下了想升級成的朋友可執(zhí)行以下代碼把之前的掉安裝現(xiàn)在的 版本:ubuntu 14.04(64位) 1.安裝 Nginx(version:1.9) 1、首先添加nginx_signing.key wget http://nginx.org/keys/ng...

    joyvw 評論0 收藏0

發(fā)表評論

0條評論

閱讀需要支付1元查看
<