PHP 大会日本 2024

CURLFile::__construct

curl_file_create

(PHP 5 >= 5.5.0, PHP 7, PHP 8)

CURLFile::__construct -- curl_file_create创建 CURLFile 对象

描述

面向对象风格

public CURLFile::__construct(字符串 $filename, ?字符串 $mime_type = null, ?字符串 $posted_filename = null)

过程化风格

curl_file_create(字符串 $filename, ?字符串 $mime_type = null, ?字符串 $posted_filename = null): CURLFile

创建 CURLFile 对象,用于使用 CURLOPT_POSTFIELDS 上传文件。

参数

filename

要上传的文件的路径。

mime_type

文件的 MIME 类型。

posted_filename

要在上传数据中使用的文件名。

返回值

返回一个 CURLFile 对象。

变更日志

版本 描述
8.0.0 mime_typeposted_filename 现在可以为空;以前它们的默认值为 0

示例

示例 #1 CURLFile::__construct() 示例

面向对象风格

<?php
/* http://example.com/upload.php:
<?php var_dump($_FILES); ?>
*/

// 创建一个 cURL句柄
$ch = curl_init('http://example.com/upload.php');

// 创建一个 CURLFile 对象
$cfile = new CURLFile('cats.jpg','image/jpeg','test_name');

// 分配 POST 数据
$data = array('test_file' => $cfile);
curl_setopt($ch, CURLOPT_POST,1);
curl_setopt($ch, CURLOPT_POSTFIELDS, $data);

// 执行句柄
curl_exec($ch);
?>

过程化风格

<?php
/* http://example.com/upload.php:
<?php var_dump($_FILES); ?>
*/

// 创建一个 cURL句柄
$ch = curl_init('http://example.com/upload.php');

// 创建一个 CURLFile 对象
$cfile = curl_file_create('cats.jpg','image/jpeg','test_name');

// 分配 POST 数据
$data = array('test_file' => $cfile);
curl_setopt($ch, CURLOPT_POST,1);
curl_setopt($ch, CURLOPT_POSTFIELDS, $data);

// 执行句柄
curl_exec($ch);
?>

以上示例将输出

array(1) {
  ["test_file"]=>
  array(5) {
    ["name"]=>
    string(9) "test_name"
    ["type"]=>
    string(10) "image/jpeg"
    ["tmp_name"]=>
    string(14) "/tmp/phpPC9Kbx"
    ["error"]=>
    int(0)
    ["size"]=>
    int(46334)
  }
}

示例 #2 CURLFile::__construct() 上传多个文件示例

面向对象风格

<?php
$request
= curl_init('http://www.example.com/upload.php');
curl_setopt($request, CURLOPT_POST, true);
curl_setopt($request, CURLOPT_SAFE_UPLOAD, true);
curl_setopt($request, CURLOPT_POSTFIELDS, [
'blob[0]' => new CURLFile(realpath('first-file.jpg'), 'image/jpeg'),
'blob[1]' => new CURLFile(realpath('second-file.txt'), 'text/plain'),
'blob[2]' => new CURLFile(realpath('third-file.exe'), 'application/octet-stream'),
]);
curl_setopt($request, CURLOPT_RETURNTRANSFER, true);

echo
curl_exec($request);

var_dump(curl_getinfo($request));

curl_close($request);

过程化风格

<?php
// 过程式
$request = curl_init('http://www.example.com/upload.php');
curl_setopt($request, CURLOPT_POST, true);
curl_setopt($request, CURLOPT_SAFE_UPLOAD, true);
curl_setopt($request, CURLOPT_POSTFIELDS, [
'blob[0]' => curl_file_create(realpath('first-file.jpg'), 'image/jpeg'),
'blob[1]' => curl_file_create(realpath('second-file.txt'), 'text/plain'),
'blob[2]' => curl_file_create(realpath('third-file.exe'), 'application/octet-stream'),
]);
curl_setopt($request, CURLOPT_RETURNTRANSFER, true);

echo
curl_exec($request);

var_dump(curl_getinfo($request));

curl_close($request);

以上示例将输出

array(26) {
  ["url"]=>
  string(31) "http://www.example.com/upload.php"
  ["content_type"]=>
  string(24) "text/html; charset=UTF-8"
  ["http_code"]=>
  int(200)
  ["header_size"]=>
  int(198)
  ["request_size"]=>
  int(196)
  ["filetime"]=>
  int(-1)
  ["ssl_verify_result"]=>
  int(0)
  ["redirect_count"]=>
  int(0)
  ["total_time"]=>
  float(0.060062)
  ["namelookup_time"]=>
  float(0.028575)
  ["connect_time"]=>
  float(0.029011)
  ["pretransfer_time"]=>
  float(0.029121)
  ["size_upload"]=>
  float(3230730)
  ["size_download"]=>
  float(811)
  ["speed_download"]=>
  float(13516)
  ["speed_upload"]=>
  float(53845500)
  ["download_content_length"]=>
  float(811)
  ["upload_content_length"]=>
  float(3230730)
  ["starttransfer_time"]=>
  float(0.030355)
  ["redirect_time"]=>
  float(0)
  ["redirect_url"]=>
  string(0) ""
  ["primary_ip"]=>
  string(13) "0.0.0.0"
  ["certinfo"]=>
  array(0) {
  }
  ["primary_port"]=>
  int(80)
  ["local_ip"]=>
  string(12) "0.0.0.0"
  ["local_port"]=>
  int(34856)
}

另请参阅

添加注释

用户贡献的注释 2 条注释

CertaiN
10 年前
在 multipart POST 请求中存在“@”问题。

PHP 5.5 或更高版本的解决方案
- 启用 CURLOPT_SAFE_UPLOAD。
- 使用 CURLFile 而不是“@”。

PHP 5.4 或更早版本的解决方案
- 自己构建 multipart 内容主体。
- 自己更改“Content-Type”标头。

以下代码片段将帮助您:D

<?php

/**
* 针对 PHP5.3 ~ PHP 5.4 的安全 multipart POST 请求。
*
* @param resource $ch cURL 资源
* @param array $assoc "名称 => 值"
* @param array $files "名称 => 路径"
* @return bool
*/
function curl_custom_postfields($ch, array $assoc = array(), array $files = array()) {

// "名称" 和 "文件名" 的无效字符
static $disallow = array("\0", "\"", "\r", "\n");

// 构建普通参数
foreach ($assoc as $k => $v) {
$k = str_replace($disallow, "_", $k);
$body[] = implode("\r\n", array(
"Content-Disposition: form-data; name=\"{$k}\"",
"",
filter_var($v),
));
}

// 构建文件参数
foreach ($files as $k => $v) {
switch (
true) {
case
false === $v = realpath(filter_var($v)):
case !
is_file($v):
case !
is_readable($v):
continue;
// 或返回 false,抛出新的 InvalidArgumentException
}
$data = file_get_contents($v);
$v = call_user_func("end", explode(DIRECTORY_SEPARATOR, $v));
$k = str_replace($disallow, "_", $k);
$v = str_replace($disallow, "_", $v);
$body[] = implode("\r\n", array(
"Content-Disposition: form-data; name=\"{$k}\"; filename=\"{$v}\"",
"Content-Type: application/octet-stream",
"",
$data,
));
}

// 生成安全边界
do {
$boundary = "---------------------" . md5(mt_rand() . microtime());
} while (
preg_grep("/{$boundary}/", $body));

// 为每个参数添加边界
array_walk($body, function (&$part) use ($boundary) {
$part = "--{$boundary}\r\n{$part}";
});

// 添加最终边界
$body[] = "--{$boundary}--";
$body[] = "";

// 设置选项
return @curl_setopt_array($ch, array(
CURLOPT_POST => true,
CURLOPT_POSTFIELDS => implode("\r\n", $body),
CURLOPT_HTTPHEADER => array(
"Expect: 100-continue",
"Content-Type: multipart/form-data; boundary={$boundary}", // 更改 Content-Type
),
));
}

?>
mipa
10 年前
对于 PHP < 5.5

<?php

if (!function_exists('curl_file_create')) {
function
curl_file_create($filename, $mimetype = '', $postname = '') {
return
"@$filename;filename="
. ($postname ?: basename($filename))
. (
$mimetype ? ";type=$mimetype" : '');
}
}

?>
To Top