오늘의 도구 소개 - CURL(클라이언트 URL 라이브러리) 물론 오늘은 이 도구를 PHP에서 사용하겠습니다.
0. 컬이 무엇인가요?
PHP supports libcurl, a library created by Daniel Stenberg, that allows you to connect and communicate to many different types of servers with many different types of protocols. libcurl currently supports the http, https, ftp, gopher, telnet, dict, file, and ldap protocols. libcurl also supports HTTPS certificates, HTTP POST, HTTP PUT, FTP uploading (this can also be done with PHP's ftp extension), HTTP form based upload, proxies, cookies, and user+password authentication.
간단히 말하면 컬은 URL과 다양한 유형의 서버를 전달할 수 있는 라이브러리입니다. 연결, 채팅, 깊은 대화 등 다양한 프로토콜을 지원합니다. 그리고 사람들은 컬이 https 인증, http 포스트, ftp 업로드, 프록시, 쿠키, 단순 비밀번호 인증 및 기타 기능을 지원할 수 있다고 말했습니다.
너무 많이 말했지만 실제로는 애플리케이션에서만 느낄 수 없었습니다. 처음에는 시작하기 전에 서버 측에서 다른 서버에 POST 요청을 시작해야 했습니다. 컬을 접하고 나서 느꼈습니다.
공식적으로 사용 방법에 대해 이야기하기 전에 먼저 PHP 환경에 컬 모듈을 설치하고 활성화해야 한다는 점을 언급하겠습니다. 시스템마다 설치 방법이 다릅니다. Google에서 확인할 수 있습니다. 확인하는 방법은 매우 간단합니다. 또는 공식 PHP 문서를 확인하는 것도 좋습니다.
1. 먼저 사용해 보세요
도구를 구입한 후 먼저 사용해 보고 자신에게 편한지 확인해야 합니다. 그렇지 않으면 즉시 사용하세요. 당신이 그것을 얻으면, 당신의 코드는 엉망이 될 것입니다. 서버를 놀리는 것은 어떻습니까?
예를 들어, 유명한 "네트워크 연결 테스트" 웹사이트인 Baidu를 예로 들어보겠습니다.
<?php // create curl resource $ch = curl_init(); // set url curl_setopt($ch, CURLOPT_URL, "baidu.com"); //return the transfer as a string curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // $output contains the output string $output = curl_exec($ch); //echo output echo $output; // close curl resource to free up system resources curl_close($ch); ?>
이 PHP 파일을 로컬 환경 브라우저에서 열면, 나타나는 페이지는 바이두 홈페이지인데, 방금 입력한 “localhost”는 어떻습니까?
위의 코드와 주석은 이 코드가 수행하는 작업을 완전히 설명했습니다.
$ch = cur_init(), 컬 세션 리소스를 생성하고 성공적으로 핸들을 반환합니다.
curl_setopt($ch, CURLOPT_URL, "baidu.com"), 말할 것도 없이 URL을 설정합니다.
위 두 문장을 한 문장으로 합칠 수 있습니다. $ch = cur_init("baidu.com");
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 0) 저장할지 여부를 설정하는 것입니다. 응답 결과는 변수 1에 저장되고, 0은 직접 에코됩니다.
$output = cur_exec($ch)가 실행된 후 다음 에코에 대한 응답 결과가 $output 변수에 저장됩니다. ;
curl_close($ ch) 이 컬 세션 리소스를 닫습니다.
PHP에서 컬을 사용하는 방법은 대략 다음과 같습니다. 두 번째 단계인 컬_setopt 메소드를 통해 매개변수를 설정하는 것이 가장 복잡하고 중요합니다. 관심이 있는 경우 설정 가능한 매개변수에 대한 공식 세부 참조를 읽어보세요. 토하고 싶을 정도로 길지만, 필요에 따라 연습하면 완벽해집니다.
요약하자면, PHP에서 컬의 사용법은 다음과 같습니다: 컬 세션 생성 -> 구성 매개변수 ->
올바른 "여자 선택"(올바른 서버 선택)을 위해 "옷을 입는 것"(구성 매개변수)이 필요한 몇 가지 일반적인 시나리오를 살펴보겠습니다.
2. Say hello - GET 및 POST 요청 및 HTTPS 프로토콜 처리
먼저 서버에 Hello를 보내고 서버가 어떻게 응답하는지 확인하는 것이 가장 편리한 방법입니다. 물론 서버에 GET 요청을 보내는 것도 괜찮습니다.
2.1 GET 요청
'유명 게이 데이트 사이트 깃허브에서 키워드 검색'을 예로 들어보겠습니다
//通过curl进行GET请求的案例 <?php // create curl resource $ch = curl_init(); // set url curl_setopt($ch, CURLOPT_URL, "https://github.com/search?q=react"); //return the transfer as a string curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // $output contains the output string $output = curl_exec($ch); //echo output echo $output; // close curl resource to free up system resources curl_close($ch); ?>
전과 별반 다를게 없는 것 같습니다 예를 들어 언급할 수 있는 두 가지 사항은 다음과 같습니다.
1. 기본 요청 방법은 GET이므로 명시적으로 GET 방법을 지정할 필요가 없습니다.
2. http가 아닌 요청의 경우 HTTPS 요청이 리소스를 성공적으로 요청하려면 SSL 인증서 확인을 우회하기 위해 몇 줄의 코드를 추가해야 한다는 것을 여러 곳에서 본 적이 있는데 여기서는 그럴 필요가 없는 것 같습니다.
The two Curl options are defined as: CURLOPT_SSL_VERIFYPEER - verify the peer's SSL certificate CURLOPT_SSL_VERIFYHOST - verify the certificate's name against host They both default to true in Curl, and shouldn't be disabled unless you've got a good reason. Disabling them is generally only needed if you're sending requests to servers with invalid or self-signed certificates, which is only usually an issue in development. Any publicly-facing site should be presenting a valid certificate, and by disabling these options you're potentially opening yourself up to security issues.
즉, 개발 환경에서 주로 발생하는 불법 또는 자체 인증서를 사용하지 않는 한 SSL 인증서 확인을 피하기 위해 이 두 줄을 false로 설정해야 하며, 그렇지 않으면 할 필요가 없습니다. 그렇게 하는 것은 안전하지 않습니다.
2.2 POST 요청
그럼 어떻게 POST 요청을 할까요? 테스트를 위해 먼저 테스트 서버에 POST를 수신하는 스크립트를 전달합니다.
//testRespond.php <?php $phpInput=file_get_contents('php://input'); echo urldecode($phpInput); ?>
일반 데이터 전송
그런 다음 로컬에서 요청을 작성합니다.
<?php $data=array( "name" => "Lei", "msg" => "Are you OK?" ); $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, "http://测试服务器的IP马赛克/testRespond.php"); curl_setopt($ch, CURLOPT_POST, 1); //The number of seconds to wait while trying to connect. Use 0 to wait indefinitely. curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 60); curl_setopt($ch, CURLOPT_POSTFIELDS , http_build_query($data)); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); $output = curl_exec($ch); echo $output; curl_close($ch); ?>
브라우저 실행 결과는 다음과 같습니다.
name=Lei&msg=Are you OK?
여기서 배열을 구성하고 POST 데이터로 서버에 전달합니다.
cur_setopt ($ch, CURLOPT_POST, 1)은 POST 요청을 나타냅니다.
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 60)는 허용 가능한 가장 긴 연결 시간을 초 단위로 설정합니다. ;
curl_setopt ($ch, CURLOPT_POSTFIELDS, http_build_query($data))는 POST의 데이터 필드를 설정합니다. 이는 배열 데이터 형식이므로(json 형식은 나중에 설명합니다.) http_build_query를 사용하여 처리합니다.
json 데이터는 어떻게 작성하나요?
<?php $data='{"name":"Lei","msg":"Are you OK?"}'; $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, "http://测试服务器的IP马赛克/testRespond.php"); curl_setopt($ch, CURLOPT_POST, 1); curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 60); curl_setopt($ch, CURLOPT_HTTPHEADER, array('Content-Type: application/json', 'Content-Length:' . strlen($data))); curl_setopt($ch, CURLOPT_POSTFIELDS , $data); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); $output = curl_exec($ch); echo $output; curl_close($ch); ?>
브라우저가 실행되고 다음이 표시됩니다.
{"name":"Lei","msg":"Are you OK?"}
3. 파일 업로드 및 다운로드 방법
은 이제 서버에 연결되어 있으므로 사진을 찍어서 다른 사람들이 볼 수 있도록 해야 합니다. 두 사람이 함께 있는 모습은 중요하지 않지만, 잘생긴 남자와 아름다운 여자는 언제나 최고입니다.
3.1 진심을 담은 사진 보내기 - POST 업로드 파일
원격 서버와 마찬가지로 먼저 수신 스크립트를 보내 사진을 받아 로컬에 저장한다. 파일 및 폴더 권한 문제, 쓰기 권한이 필요합니다:
<?php if($_FILES){ $filename = $_FILES['upload']['name']; $tmpname = $_FILES['upload']['tmp_name']; //保存图片到当前脚本所在目录 if(move_uploaded_file($tmpname,dirname(__FILE__).'/'.$filename)){ echo ('上传成功'); } } ?>
然后我们再来写我们本地服务器的php curl部分:
<?php $data = array('name'=>'boy', "upload"=>"@boy.png"); $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, "http://远程服务器地址马赛克/testRespond.php"); curl_setopt($ch, CURLOPT_POST, 1); curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 60); curl_setopt($ch, CURLOPT_POSTFIELDS , $data); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); $output = curl_exec($ch); echo $output; curl_close($ch); ?>
浏览器中运行一下,什么都米有,去看一眼远程的服务器,还是什么都没有,并没有上传成功。
为什么会这样呢?上面的代码应该是大家搜索curl php POST图片最常见的代码,这是因为我现在用的是PHP5.6以上版本,@符号在PHP5.6之后就弃用了,PHP5.3依旧可以用,所以有些同学发现能执行啊,有些发现不能执行,大抵是因为PHP版本的不同,而且curl在这两版本中实现是不兼容的,上面是PHP5.3的实现。
下面来讲PHP5.6及以后的实现,:
<?php $data = array('name'=>'boy', "upload"=>""); $ch = curl_init(); $data['upload']=new CURLFile(realpath(getcwd().'/boy.png')); curl_setopt($ch, CURLOPT_URL, "http://115.29.247.189/test/testRespond.php"); curl_setopt($ch, CURLOPT_POST, 1); curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 60); curl_setopt($ch, CURLOPT_POSTFIELDS , $data); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); $output = curl_exec($ch); echo $output; curl_close($ch); ?>
这里引入了一个CURLFile对象进行实现,关于此的具体可查阅文档进行了解。这时候再去远程服务器目录下看看,发现有了一张图片了,而且确实是我们刚才上传的图片。
3.2 获取远程服务器妹子的照片 —— 抓取图片
服务器妹子也挺实诚的,看了照骗觉得我长得挺慈眉善目的,就大方得拿出了她自己的照片,但是有点害羞的是,她不愿意主动拿过来,得我们自己去取。
远程服务器在她自己的目录下存放了一个图片叫girl.jpg,地址是她的web服务器根目录/girl.jpg,现在我要去获取这张照片。
<?php $ch = curl_init(); $fp=fopen('./girl.jpg', 'w'); curl_setopt($ch, CURLOPT_URL, "http://远程服务器地址马赛克/girl.jpg"); curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 60); curl_setopt($ch, CURLOPT_FILE, $fp); $output = curl_exec($ch); $info = curl_getinfo($ch); fclose($fp); $size = filesize("./girl.jpg"); if ($size != $info['size_download']) { echo "下载的数据不完整,请重新下载"; } else { echo "下载数据完整"; } curl_close($ch); ?>
现在,在我们当前目录下就有了一张刚拿到的照片啦,是不是很激动呢!
这里值得一说的是curl_getinfo方法,这是一个获取本次请求相关信息的方法,对于调试很有帮助,要善用。
4. HTTP认证怎么搞
这个时候呢,服务器的家长说这个我们女儿还太小,不能找对象,就将她女儿关了起来,并且上了一个密码锁,所谓的HTTP认证,服务器呢偷偷托信鸽将HTTP认证的用户名和密码给了你,要你去见她,带她私奔。
那么拿到了用户名和密码,我们怎么通过PHP CURL搞定HTTP认证呢?
PS:这里偷懒就不去搭HTTP认证去试了,直接放一段代码,我们分析下。
function curl_auth($url,$user,$passwd){ $ch = curl_init(); curl_setopt_array($ch, [ CURLOPT_USERPWD => $user.':'.$passwd, CURLOPT_URL => $url, CURLOPT_RETURNTRANSFER => true ]); $result = curl_exec($ch); curl_close($ch); return $result; } $authurl = 'http://要请求HTTP认证的地址'; echo curl_auth($authurl,'vace','passwd');
这里有一个地方比较有意思: curl_setopt_array 这个方法可以通过数组一次性地设置多个参数,防止有些需要多处设置的出现密密麻麻的curl_setopt方法。 5.利用cookie模拟登陆 这时你成功见到了服务器妹子,想带她私奔,但是无奈没有盘缠走不远,服务器妹子说,她妈服务器上有金库,可以登陆上去搞一点下来。 首先我们先来分析一下,这个事情分两步,一是去登陆界面通过账号密码登陆,然后获取cookie,二是去利用cookie模拟登陆到信息页面获取信息,大致的框架是这样的。
<?php //设置post的数据 $post = array ( 'email' => '账户', 'pwd' => '密码' ); //登录地址 $url = "登陆地址"; //设置cookie保存路径 $cookie = dirname(__FILE__) . '/cookie.txt'; //登录后要获取信息的地址 $url2 = "登陆后要获取信息的地址"; //模拟登录 login_post($url, $cookie, $post); //获取登录页的信息 $content = get_content($url2, $cookie); //删除cookie文件 @ unlink($cookie); var_dump($content); ?>
然后我们思考下下面两个方法的实现:
login_post($url, $cookie, $post) get_content($url2, $cookie) //模拟登录 function login_post($url, $cookie, $post) { $curl = curl_init(); curl_setopt($curl, CURLOPT_URL, $url); curl_setopt($curl, CURLOPT_RETURNTRANSFER, 0); curl_setopt($curl, CURLOPT_COOKIEJAR, $cookie); curl_setopt($curl, CURLOPT_POST, 1); curl_setopt($curl, CURLOPT_POSTFIELDS, http_build_query($post)); curl_exec($curl); curl_close($curl); } //登录成功后获取数据 function get_content($url, $cookie) { $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_HEADER, 0); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); curl_setopt($ch, CURLOPT_COOKIEFILE, $cookie); $rs = curl_exec($ch); curl_close($ch); return $rs; }
至此,总算是模拟登陆成功,一切顺利啦,通过php CURL“撩”服务器就是这么简单。
当然,CURL的能力远不止于此,本文仅希望就后端PHP开发中最常用的几种场景做一个整理和归纳。最后一句话,具体问题具体分析。
更多php使用curl详细解析及问题汇总相关文章请关注PHP中文网!