PHP 模拟 POST 请求的三种方式

有时候写 PHP 项目,需要在服务端主动发起 POST 请求——比如调用第三方接口、模拟表单提交、做个简单的采集之类的。这种需求还挺常见的,我这里整理了三种常用方式,备忘一下。

一、cURL(最推荐)

cURL 是用得最多的方式,功能强、灵活,基本上能处理各种情况:

1
2
3
4
5
6
7
8
9
10
11
function request_by_curl($remote_server, $post_string)
{
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, $remote_server);
curl_setopt($ch, CURLOPT_POSTFIELDS, 'mypost=' . $post_string);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_USERAGENT, "Jimmy's CURL Example beta");
$data = curl_exec($ch);
curl_close($ch);
return $data;
}

用法很简单:

1
2
$post_string = "app=request&version=beta";
$result = request_by_curl('http://example.com/api.php', $post_string);

基本上能覆盖 90% 的使用场景,支持 HTTPS、设置 Header、处理 Cookie 也都方便。

二、file_get_contents + stream_context

如果服务器没装 cURL 扩展,可以用这个方案,PHP 原生支持,不需要额外扩展:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
function request_by_other($remote_server, $post_string)
{
$context = array(
'http' => array(
'method' => 'POST',
'header' => 'Content-type: application/x-www-form-urlencoded' .
"\r\n" . 'Content-length: ' . (strlen($post_string) + 8),
'content' => 'mypost=' . $post_string
)
);
$stream_context = stream_context_create($context);
$data = file_get_contents($remote_server, false, $stream_context);
return $data;
}

用法:

1
2
$post_string = "app=request&version=beta";
$result = request_by_other('http://example.com/api.php', $post_string);

缺点是不支持 HTTPS 证书验证配置没那么方便,复杂场景还是老老实实用 cURL。

三、Socket 原生方式

这种方式是最底层的,直接用 fsockopen 建立 TCP 连接,手动拼 HTTP 报文:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
function request_by_socket($remote_server, $remote_path, $post_string, $port = 80, $timeout = 30)
{
$socket = fsockopen($remote_server, $port, $errno, $errstr, $timeout);
if (!$socket) die("$errstr($errno)");

fwrite($socket, "POST $remote_path HTTP/1.0\r\n");
fwrite($socket, "HOST: $remote_server\r\n");
fwrite($socket, "Content-type: application/x-www-form-urlencoded\r\n");
fwrite($socket, "Content-length: " . (strlen($post_string) + 8) . "\r\n");
fwrite($socket, "Accept: */*\r\n");
fwrite($socket, "\r\n");
fwrite($socket, "mypost=$post_string\r\n");
fwrite($socket, "\r\n");

$data = "";
// 跳过响应头
while ($str = trim(fgets($socket, 4096))) {}
// 读响应体
while (!feof($socket)) {
$data .= fgets($socket, 4096);
}
fclose($socket);
return $data;
}

用法:

1
2
$post_string = "app=socket&version=beta";
$result = request_by_socket('example.com', '/api.php', $post_string);

日常开发基本用不上这个,太麻烦了。但如果你想搞清楚 HTTP 底层是怎么工作的,自己手写一遍还挺有意思的。

有时候需要模拟登录然后同步 Cookie,比如做多个系统之间的单点登录,可以这样处理响应里的 Set-Cookie

1
2
3
4
5
6
7
8
9
10
11
while (!feof($fp)) {
$str = fgets($fp);
if (strpos($str, "Set-Cookie:") !== false) {
$tmparray = explode(" ", $str);
$cookiearray = explode("=", $tmparray[1]);
$cookiename = urldecode($cookiearray[0]);
$cookievalue = urldecode(rtrim($cookiearray[1]));
$cookietime = time() + 3600 * 24 * 7; // 保存7天
setcookie($cookiename, $cookievalue, $cookietime, '/');
}
}

把服务端返回的 Cookie 解析出来再 setcookie 写到当前用户的浏览器,就实现了同步登录状态的效果。


总结一下:日常用 cURL,没有 cURL 用 file_get_contents,想玩底层用 Socket。希望对遇到同样需求的小伙伴有帮助~