激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - PHP教程 - PHP5.0~5.6 各版本兼容性cURL文件上傳功能實例分析

PHP5.0~5.6 各版本兼容性cURL文件上傳功能實例分析

2019-10-08 10:23沙渺 PHP教程

這篇文章主要介紹了PHP5.0~5.6 各版本兼容性cURL文件上傳功能,結合實例形式分析了php各個常見版本進行curl文件上傳操作的相關實現技巧與注意事項,需要的朋友可以參考下

本文實例分析了PHP5.0~5.6 各版本兼容性cURL文件上傳功能。分享給大家供大家參考,具體如下:

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

重要警告

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

不同版本PHP之間cURL的區別

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

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

curl_setopt(ch, CURLOPT_POSTFIELDS, array(
  'file' => '@'.realpath('image.png'),
));

equals

$ curl -F "file=@/absolute/path/to/image.png" <url>

但PHP從5.5開始引入了新的CURLFile類用來指向文件。CURLFile類也可以詳細定義MIME類型、文件名等可能出現在multipart/form-data數據中的附加信息。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就已經被打了deprecated,在5.6中就直接被刪除了(會產生 ErorException: The usage of the @filename API for file uploading is deprecated. Please use the CURLFile class instead)。

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

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

現在問題來了……

環境判斷:小心魔法數字!

我見過這種環境判斷的代碼:

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

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

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

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

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選項設置的順序

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

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

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

實際上在curl_setopt_array()用的數組中,保證CURLOPT_POSTFIELDS的位置在后邊也是可靠的。PHP的關聯數組是有順序保障的,我們也可以假設curl_setopt_array()內部的執行順序一定是從頭到尾按順序(好吧我知道assume不是件好事,不過有些實在過分淺顯的事實,就容我下個最低限度的斷言吧),所以盡可放心。

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

命名空間

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

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

希望本文所述對大家PHP程序設計有所幫助。

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 亚洲最黄视频 | 中国女人内谢69xxxx天美 | 精品一区二区三区免费毛片 | 美女黄色毛片免费看 | 黄色免费av网站 | 一区二区三区四区五区中文字幕 | 狠狠操夜夜爱 | 免费看性xxx高清视频自由 | 国产一区免费在线 | 精品国产91一区二区三区 | 亚洲午夜精品视频 | 特片网久久 | 国产色视频一区 | av影院在线播放 | 成人精品免费在线观看 | 色玖玖综合 | 国产精品视频中文字幕 | 亚洲性在线视频 | 911色_911色sss主站色播 | 一区二区三区在线观看国产 | 欧美性激情视频 | 成人黄色免费电影 | 日本一区二区三区视频在线 | 日本一区二区免费在线观看 | 久草在线资源观看 | 巨根插入 | av免费大全 | 亚洲不卡| 性生活视频一级 | 国产一级片91 | 免费看成人av | 一级片久久免费 | 精品久久久久久久久久久aⅴ | 无码专区aaaaaa免费视频 | 久久久久久久久日本理论电影 | 污视频在线免费 | 天天鲁在线视频免费观看 | 色视频一区二区 | 毛片国产 | 91情侣偷在线精品国产 | 成人资源在线观看 |