起先cURL是做为一种命令行工具设计出来的,比较幸运的是,php也支持cURL了。通过cURL这个利器,我们能在php程序中*地发送 HTTP请求到某个url来获取或者提交数据,并且支持其它多种协议,比如FTP,Telnet以及SMTP等。在这篇博文中,我将简述下,在php中具 体怎么使用cURL来处理一些事情。
cURL的优势
你也许会说,在php中可以很容易的获取某个url的内容,只要通过file_get_contents,file或者readfile函数就能轻松实现,根本不必使用cURL:
1 $content = file_get_contents("http://www.360weboy.com"); 2 $lines = file("http://www.360weboy.com"); 3 readfile("http://www.360weboy.com");
没错,以上函数在某些情况下使用起来确实很方便,但是我感觉这几个函数不够灵活,也没法进行错误处理。而且,如果遇到要在php程序中向某个服务器 提交表单数据,上传文件,处理cookies或者认证等任务时,以上三个函数根本无法胜任。这个时候,cURL就体现它的价值了。
cURl不但支持很多的网络协议,而且提供了关于url请求的具体信息,很强大!
cURL的简单使用步骤
要使用cURL来发送url请求,具体步骤大体分为以下四步:
- 初始化
- 设置请求选项
- 执行一个cURL会话并且获取相关回复
- 释放cURL句柄,关闭一个cURL会话
1 // 1. 初始化一个cURL会话 2 $ch = curl_init(); 3 4 // 2. 设置请求选项, 包括具体的url 5 curl_setopt($ch, CURLOPT_URL, "http://www.360weboy.com"); 6 curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); 7 curl_setopt($ch, CURLOPT_HEADER, 0); 8 9 // 3. 执行一个cURL会话并且获取相关回复 10 $response = curl_exec($ch); 11 12 // 4. 释放cURL句柄,关闭一个cURL会话 13 curl_close($ch);
cURL之所以强大,正是体现在第二个步骤中。你可以通过curl_setopt灵活地设置请求选项,这里面有很多的可选项,具体可以参考http://cn2.php.net/manual/zh/function.curl-setopt.php
错误处理
在上述代码中,你也可以增加错误处理的代码:
$response = curl_exec($ch); if ($response === FALSE) { echo "cURL 具体出错信息: " . curl_error($ch); }
注意了,在做上述判断时务必要使用===,因为请求的回复可能是空字符串,curl在请求出错的情况下回返回FALSE值,所以我们必须使用===,而不是==。
获取cURL请求的具体信息
在执行一个cURL请求后,你也可以使用curl_getinfo获取该请求的具体信息:
1
2
3
4
|
curl_exec( $ch );
$curl_info = curl_getinfo( $ch );
echo
"收到的http回复的code为: {$curl_info[‘http_code‘]}" ;
|
上述$curl_info是一个关联数组,可以从中获取很多的具体请求信息。参考http://cn2.php.net/manual/zh/function.curl-getinfo.php
使用cURL发送Post请求
我们在前面说过,在向某个url发送get请求的话,没有必要使用cURL来发送get请求,可以使用比较便捷的 file_get_contents函数来完成请求。但是,一般地,我们在提交某个表单的时候,数据是通过post请求的内容区域来提交的,而不是通过 url参数来传递的, 这种情况下,我们应该使用灵活的cURL来模拟发送post请求。
现在,让我们使用cURL来模拟发送一个post请求到post.php脚本,提交几个数据到post.php,然后在post.php中输出post请求中的数据。示例代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
$post_data
= array
(
"blog_name"
=> "360weboy" ,
"action"
=> "Submit"
); $ch
= curl_init();
curl_setopt( $ch , CURLOPT_URL, $url );
curl_setopt( $ch , CURLOPT_RETURNTRANSFER, 1);
// 设置请求为post类型 curl_setopt( $ch , CURLOPT_POST, 1);
// 添加post数据到请求中 curl_setopt( $ch , CURLOPT_POSTFIELDS, $post_data );
// 执行post请求,获得回复 $response = curl_exec( $ch );
curl_close( $ch );
echo
$response ;
|
以上请求发送到post.php中后,通过print_r($_POST)输出后,以上示例代码会输出如下回复:
1
2
3
4
5
6
|
Array ( [blog_name] => 360weboy
[blog_url] => http: //www.360weboy.com
[action] => Submit
) |
正如我们看到的,cURL成功发送post请求到post.php,提交了一些数据,并且收到了相应的来自post.php的回复,最后输出回复。上例虽然简单,但是充分演示了cURL发送post请求的便捷及强大之处,你可以在curl_setopt上做文章。
文件上传
下面来看下如果通过cURL发送post请求来实现文件上传。就拿深入浅出PHP下的文件上传中的文件上传例子来演示,在深入浅出php下的文件上传中,是通过表单的提交来实现文件上传的,那么通过cURL怎么来实现呢?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
$post_data
= array
(
"attachment"
=> "@E:/jackblog/boy.jpg"
); //初始化cURL会话 $ch
= curl_init();
//设置请求的url curl_setopt( $ch , CURLOPT_URL, $url );
curl_setopt( $ch , CURLOPT_RETURNTRANSFER, 1);
//设置为post请求类型 curl_setopt( $ch , CURLOPT_POST, 1);
//设置具体的post数据 curl_setopt( $ch , CURLOPT_POSTFIELDS, $post_data );
$response
= curl_exec( $ch );
curl_close( $ch );
print_r( $response );
|
通过以上示例代码,可以将我本地机器上的boy.jpg上传到本地服务器的upload.php中,如果在upload.php输出上传的具体信息的话,以上示例代码最后的输出的回复为:
1
2
3
4
5
6
7
8
9
10
11
12
|
Array ( [attachment] => Array
(
[name] => boy.jpg
[type] => application/octet-stream
[tmp_name] => D:\xampp\tmp\phpF27D.tmp
[error] => 0
[size] => 11490
)
) |
由此可见,如果你要通过cURL来上传文件的话,只需要将上传的文件路径作为post数据设置到curl请求中,并且在路径前面加上@符合。
文件下载
上述将了文件上传,同样的也可以使用curl来自动地完成文件的下载以及保存。有一点要补充下,在执行一个curl请求时,如果你需要获取返回的内容,而不是直接输出返回的内容的话,别忘记使用下面的代码设置,因为curl的默认是输出请求的回复内容:
1
|
curl_setopt( $ch , CURLOPT_RETURNTRANSFER, 1);
|
假如在360weboy的服务器根目录下面有一个test.zip文件,我们需要将其下载下来,并且保存到本地文件中,就可以尝试使用下面的代码来实现:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
//设置请求的下载文件的url //保存到本地的文件路径 $path
= ‘local/path/to/test.zip‘ ;
//初始化请求,设置请求,获取回复,关闭会话 $ch
= curl_init( $url );
curl_setopt( $ch , CURLOPT_RETURNTRANSFER, true);
$data
= curl_exec( $ch );
curl_close( $ch );
//将文件内容写入本地文件 file_put_contents ( $path , $data );
|
注意:我以上省略了错误处理方面的代码,只是简单做个示例, 在实际中,你还需要通过curl_getinfo函数来进行错误处理!
上述代码对于下载比较大型的文件是不适用的,因为需要先将文件读取到内存中,等所有内容都读取完毕,然后再写入到本地硬盘中。即使php中设置的 memory limit非常大,这种情况对性能的影响也是很大的。所以,我们对于大型文件的下载,应该让curl来接管这个任务,实现边下载,边写入的处理,这样的 话,就没什么问题了。请看下述代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
$path
= ‘local/path/to/test.zip‘ ;
// 打开本地文件 $fp
= fopen ( $path , ‘w‘ );
// 告诉curl本地文件句柄 $ch
= curl_init( $url );
curl_setopt( $ch , CURLOPT_FILE, $fp );
curl_exec( $ch );
curl_close( $ch );
fclose( $fp );
|
在上述代码中,我们先打开个本地文件,并将文件句柄设置到curl中,然后让curl一边读取远程数据,一边写入到本地文件中。因为我们不需要在程序中获取远程回复的内容了,所以只要执行请求就可以。
HTTP 验证
如果服务器端需要验证请求,可以通过类似一下示例代码来实现:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
$ch
= curl_init();
curl_setopt( $ch , CURLOPT_URL, $url );
curl_setopt( $ch , CURLOPT_RETURNTRANSFER, 1);
// 设置用户名以及密码 curl_setopt( $ch , CURLOPT_USERPWD, "username:password" );
// 设置重导向 curl_setopt( $ch , CURLOPT_FOLLOWLOCATION, 1);
curl_setopt( $ch , CURLOPT_UNRESTRICTED_AUTH, 1);
$response
= curl_exec( $ch );
curl_close( $ch );
|
通过代理发送请求
cURL还可以通过代理服务器来向发送请求,请看一下示例代码:
1
2
3
4
5
6
7
8
9
10
11
12
|
$ch
= curl_init();
curl_setopt( $ch , CURLOPT_RETURNTRANSFER, 1);
// 设置代理ip地址 curl_setopt( $ch , CURLOPT_PROXY, ‘222.73.173.50:8080‘ );
// 要验证的话,这里设置用户名以及密码 curl_setopt( $ch , CURLOPT_PROXYUSERPWD, ‘username:password‘ );
$response
= curl_exec( $ch );
curl_close ( $ch );
|
发送json数据
最后,我们来看下通过cURL来想服务器端发送json数据。具体的代码如下:
$url = ‘http://www.360weboy.me/json.php‘; // 建立json字符串 $data = array(‘site‘ => ‘360weboy‘, ‘url‘ => ‘http://www.360weboy.com‘,‘email‘=>‘360weboy@gmail.com‘); $json_string = json_encode($data); $ch=curl_init($url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // 通过post请求发送上述json字符串 curl_setopt($ch, CURLOPT_CUSTOMREQUEST, "POST"); curl_setopt($ch, CURLOPT_POSTFIELDS, array(‘data‘=>$json_string)); $response = curl_exec($ch); curl_close($ch); echo $response;
大家可以看到,上述请求是发送到我的本地服务器的json.php下,我在该文件中使用json_decode来将接受到的json字符串转换为对象,然后输出其中的email字段,代码如下:
$json_data = json_decode($_POST[‘data‘]); echo $json_data->email;
在上述代码中接受的json字符串为:
1 ‘{"site":"360weboy","url":"http:\/\/www.360weboy.com","email":"360weboy@gmail.com"}‘
经过json_decode以后,就转换为php中的数据格式,成为了一个对象,所以可以通过$json_data->email来访问其中email字段的值,最后也就是输出360weboy@gmail.com。你可以使用上述代码测试一下。
如果通过以下php数组生成json字符串的话:
1 $data = array(‘360weboy‘, ‘http://www.360weboy.com‘, ‘360weboy@gmail.com‘);
所生成的json字符串如下:
1 ‘["360weboy","http:\/\/www.360weboy.com","360weboy@gmail.com"]‘
上述json字符串在经过json_decode处理后,就会变成php中的数组格式,如果要获取email的话,就可以通过$json_data[2]来访问。