웹 서버는 PHP 스크립트를 실행하는데, 때로는 실행 결과를 반환하는 데 오랜 시간이 걸리고, 후속 스크립트가 계속 실행되기까지 오랜 시간을 기다려야 합니다. 실행 결과를 기다리지 않고 시간이 많이 걸리는 스크립트의 실행을 간단히 트리거한 후 다음 작업을 직접 수행하려는 경우 fscokopen 함수를 사용하면 됩니다.
PHP는 소켓 프로그래밍을 지원합니다. fscokopen 함수는 fopen에서 반환된 핸들을 사용하는 것과 마찬가지로 fwrite, fgets, fread 및 기타 작업을 수행할 수 있습니다. fsockopen을 사용하여 로컬 서버에 연결하고 스크립트 실행을 트리거한 다음 스크립트 실행이 완료될 때까지 기다리지 않고 즉시 반환하여 비동기 PHP 실행 효과를 얻습니다.
<? function triggerRequest($url, $post_data = array(), $cookie = array()){ $method = "GET"; //通过POST或者GET传递一些参数给要触发的脚本 $url_array = parse_url($url); //获取URL信息 $port = isset($url_array['port'])? $url_array['port'] : 80; $fp = fsockopen($url_array['host'], $port, $errno, $errstr, 30); if (!$fp) { return FALSE; } $getPath = $url_array['path'] ."?". $url_array['query']; if(!empty($post_data)){ $method = "POST"; } $header = $method . " " . $getPath; $header .= " HTTP/1.1\r\n"; $header .= "Host: ". $url_array['host'] . "\r\n "; //HTTP 1.1 Host域不能省略 /*以下头信息域可以省略 $header .= "User-Agent: Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.8.1.13) Gecko/20080311 Firefox/2.0.0.13 \r\n"; $header .= "Accept: text/xml,application/xml,application/xhtml+xml,text/html;q=0.9,text/plain;q=0.8,image/png,q=0.5 \r\n"; $header .= "Accept-Language: en-us,en;q=0.5 "; $header .= "Accept-Encoding: gzip,deflate\r\n"; */ $header .= "Connection:Close\r\n"; if(!empty($cookie)){ $_cookie = strval(NULL); foreach($cookie as $k => $v){ $_cookie .= $k."=".$v."; "; } $cookie_str = "Cookie: " . base64_encode($_cookie) ." \r\n"; //传递Cookie $header .= $cookie_str; } if(!empty($post_data)){ $_post = strval(NULL); foreach($post_data as $k => $v){ $_post .= $k."=".$v."&"; } $post_str = "Content-Type: application/x-www-form-urlencoded\r\n"; $post_str .= "Content-Length: ". strlen($_post) ." \r\n"; //POST数据的长度 $post_str .= $_post."\r\n\r\n "; //传递POST数据 $header .= $post_str; } fwrite($fp, $header); //echo fread($fp, 1024); //服务器返回 fclose($fp); return true; }
이러한 방식으로 fsockopen() 함수를 사용하여 PHP 스크립트 실행을 트리거한 다음 함수가 반환됩니다. 그런 다음 다음 단계로 진행하십시오.
현재 문제가 있습니다. 클라이언트 연결이 끊어지면, 즉 TriggerRequest가 요청을 보낸 후 연결이 즉시 닫혀 서버에서 실행 중인 스크립트가 종료될 수 있습니다.
PHP 내부에서, 시스템은 연결 상태를 유지하며 해당 상태에는 세 가지 가능한 상황이 있습니다.
* 0 - NORMAL(정상)
* 1 - ABORTED(비정상 종료)
* 2 - TIMEOUT(시간 초과)
PHP 스크립트 NORMAL 상태에서 정상적으로 동작하면 연결이 유효한 것입니다. 클라이언트 연결이 끊어지면 ABORTED 상태 플래그가 켜집니다. 원격 클라이언트 연결 중단은 일반적으로 사용자가 중지 버튼을 클릭하면 발생합니다. 연결 시간이 PHP의 시간 제한(set_time_limit() 함수 참조)을 초과하면 TIMEOUT 상태 플래그가 켜집니다.
은 클라이언트 연결이 끊어질 때 스크립트를 종료해야 하는지 여부를 결정할 수 있습니다. 어떤 원격 브라우저도 스크립트의 출력을 받아들이지 않더라도 스크립트를 완전히 실행하는 것이 편리한 경우가 있습니다. 기본값은 원격 클라이언트 연결이 중단되면 스크립트가 종료되는 것입니다. 이 처리는 php.ini의ignore_user_abort 또는 Apache .conf 설정의 해당 "php_valueignore_user_abort" 및ignore_user_abort() 함수에 의해 제어될 수 있습니다. PHP가 사용자 중단을 무시하지 않으면, 실행이 종료될 때 호출될 수 있는 다른 함수를 설정할 수 있는 Register_shutdown_function()을 제외하고 스크립트가 중단됩니다. 이는 스크립트가 실행을 완료하거나 예기치 않게 종료될 때를 의미합니다. 실행이 종료되려고 하면 함수가 호출됩니다. 원격 사용자가 STOP 버튼을 클릭하고 스크립트가 데이터를 다시 출력하려고 하면 PHP는 연결이 중단되었음을 감지하고 종료 트리거 함수를 호출합니다.
내장된 스크립트 타이머로 인해 스크립트가 중단될 수도 있습니다. 기본 시간 초과 제한은 30초입니다. 이 값은 php.ini에서 max_execution_time을 설정하거나 Apache .conf 설정 또는 set_time_limit() 함수에서 해당 "php_value max_execution_time" 매개변수를 설정하여 변경할 수 있습니다. 카운터 시간이 초과되면 위의 연결 중단 상황과 유사하게 스크립트가 종료되며, 이때 이전에 등록된 종료 트리거 함수도 실행됩니다. 종료 트리거 함수에서는 Connection_status() 함수를 호출하여 타임아웃으로 인해 종료 트리거 함수가 호출되었는지 확인할 수 있습니다. 시간 초과로 인해 종료 트리거 함수가 호출되면 함수는 2를 반환합니다.
주목해야 할 점은 ABORTED 및 TIMEOUT 상태가 동시에 유효할 수 있다는 것입니다. 이는 PHP가 사용자 종료 작업을 무시하도록 지시할 때 가능합니다. PHP는 사용자의 연결이 끊어졌지만 스크립트는 계속 실행 중임을 계속 알 수 있습니다. 실행 시간 제한에 도달하면 스크립트가 종료되고 설정된 종료 트리거 기능도 실행됩니다. 이 시점에서 연결_상태() 함수가 3을 반환한다는 것을 알 수 있습니다.
따라서 트리거할 스크립트에
<?php ignore_user_abort(TRUE); //如果客户端断开连接,不会引起脚本abort set_time_limit(0); //取消脚本执行延时上限
을 지정하거나
<?php register_shutdown_function(callback fuction[, parameters]); //注册脚本退出时执行的函数