首先multipartContent-Type的第一条规则是定义一个边界,作为每个部分之间的分隔符(因为顾名思义,它可以有多个部分)。边界可以是任何不包含在内容正文中的字符串。我通常会使用时间戳:
define('MULTIPART_BOUNDARY', '--------------------------'.microtime(true));
一旦定义了边界,您必须将其与Content-Type 标头一起发送,以告知网络服务器期望的分隔符:
$header = 'Content-Type: multipart/form-data; boundary='.MULTIPART_BOUNDARY;
完成后,您必须构建与 HTTP 规范和您发送的标头相匹配的适当内容主体。如您所知,当从表单 POST 文件时,您通常会有一个表单字段名称。我们将定义它:
// equivalent to <input type="file" name="uploaded_file"/>
define('FORM_FIELD', 'uploaded_file');
然后我们构建内容体:
$filename = "/path/to/uploaded/file.zip";
$file_contents = file_get_contents($filename);
$content = "--".MULTIPART_BOUNDARY."\r\n".
"Content-Disposition: form-data; name=\"".FORM_FIELD."\"; filename=\"".basename($filename)."\"\r\n".
"Content-Type: application/zip\r\n\r\n".
$file_contents."\r\n";
// add some POST fields to the request too: $_POST['foo'] = 'bar'
$content .= "--".MULTIPART_BOUNDARY."\r\n".
"Content-Disposition: form-data; name=\"foo\"\r\n\r\n".
"bar\r\n";
// signal end of request (note the trailing "--")
$content .= "--".MULTIPART_BOUNDARY."--\r\n";
如您所见,我们发送带有form-data 处置的Content-Disposition 标头,以及name 参数(表单字段名称)和filename 参数(原始文件名)。如果您想正确填充 $_FILES[]['type'] 事物,发送具有正确 MIME 类型的 Content-Type 标头也很重要。
如果您要上传多个文件,您只需使用 $content 位重复该过程,当然,每个文件使用不同的FORM_FIELD。
现在,构建上下文:
$context = stream_context_create(array(
'http' => array(
'method' => 'POST',
'header' => $header,
'content' => $content,
)
));
并执行:
file_get_contents('http://url/to/upload/handler', false, $context);
注意:在发送二进制文件之前无需对其进行编码。 HTTP 可以很好地处理二进制文件。