>백엔드 개발 >PHP 튜토리얼 >멀티스레드 cURL

멀티스레드 cURL

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB원래의
2016-07-25 08:49:341039검색
이는 cURL에 대한 유연한 다중 스레드 호출입니다.
PHP 매뉴얼 http://us2.php.net/manual/zh/function.curl-multi-select.php 에서 제공하는 샘플과 다르며, 코드 실행 효율이 훨씬 높습니다
여기에 두 가지 파일이 있습니다. 하나는 두 개의 클래스를 포함하는 muti_curl 파일입니다
一个是运사용 가능한 방법, 这里是批weight检查代理ip是否可用
  1. class request_setting {
  2. public $url = false;
  3. public $method = 'GET';
  4. public $ post_data = null;
  5. 공개 $headers = null;
  6. 공개 $options = null;
  7. 함수 __construct($url, $method = "GET", $post_data = null, $headers = null, $options = null) {
  8. $this->url = $url;
  9. $this->method = $method;
  10. $this->post_data = $post_data;
  11. $this-> ;headers = $headers;
  12. $this->options = $options;
  13. }
  14. 공개 함수 __destruct() {
  15. unset($this->url, $this->method , $this->post_data, $this->headers, $this->options);
  16. }
  17. }
  18. /**************************************************** * *****************************************
  19. 일괄 작업 클래스
  20. ************************************************ **** ********************************************/
  21. class muti_curl {
  22. protected $thread_size = 100;
  23. protected $timeout = 30;
  24. private $callback;
  25. protected $options = array(
  26. CURLOPT_SSL_VERIFYPEER => false,//禁用后cURL将终止从服务端进行验证。使用 CURLOPT_CAINFO选项设置证书使用 CURLOPT_CAPATH选项设置证书目录 如果CURLOPT_SSL_VERIFYPEER(默认值为2)被启用CURLOPT_SSL_VERIFYHOST需要被设置成TRUE否则设置为FALSE。 自cURL 7.10开始默认为TRUE。从cURL 7.10开始默认绑 정 안정 。
  27. CURLOPT_RETURNTRANSFER => true, //CURLOPT_CONNECTTIMEOUT => 15,
  28. CURLOPT_TIMEOUT => 30,
  29. // CURLOPT_HTTP_VERSION=>CURL_HTTP_VERSION_1_0, //사용하는 현대적인 时候用这个去抓取数据,更爽
  30. // CURLOPT_AUTOREFERER=>false,// :결정중점,자체 헤더 중 추천자: 信息.
  31. // CURLOPT_BINARYTRANSFER=>false, //현재 CURLOPT_RETURNTRANSFER의 时候, 返回原生的(Raw)이 출력되지 않습니다.
  32. // ESSION=>사실,/ / 启用时curl会仅仅传递一个session cookie,忽略其他的cookie,默认状况下cURL会将所有的cookie返回
  33. // CURLOPT_CRLF=>false,// 启用时将Unix의换行符转换成回车换行符.
  34. // CURLOPT_DNS_USE_GLOBAL_CACHE=>false, // 启用时会启用一个全局的DNS缓存,此项为线程보안에 사용됩니다.
  35. // FAILONERROR=>false, //显示HTTP状态码,默认行为是忽略编号小于等于400的HTTP信息。
  36. // CURLOPT_FILETIME=>true, //启用时会尝试修改远程文档中당신은 매우 만족스럽습니다. CURLINFO_FILETIME이 실행되었습니다.curl_getinfo().
  37. // CURLOPT_FOLLOWLOCATION=>false, // 启用时会将服务器服务器返回的"위치: "放在header中递归的返回给服务器,使用CURLOPT_MAXREDIRS可以递归返回의 양이 결정됩니다.
  38. // CURLOPT_FORBID_REUSE =>true, //현재는 사용하지 않습니다.
  39. // CURLOPT_FRESH_CONNECT=>true,//
  40. // CURLOPT_FTP_USE_EPRT=>false,// 启用时当FTP下载时,使用EPRT (或 LPRT)命令。设置为FALSE时禁用EPRT와LPRT,使用PORT命令만 가능합니다.
  41. // CURLOPT_FTP_USE_EPSV=>false,// FTP传输过程中回复到PASV模式前首先尝试EPSV命令。设置为FALSE时禁用EPSV命令。
  42. // CURLOPT_FTPAPPEND=>false, // 사용 가능한 时追加写入文件而不是覆盖它.
  43. // CURLOPT_FTPASCII=>false,// CURLOPT_TRANSFERTEXT적 이름.
  44. // CURLOPT_FTPLISTONLY=>false,// 사용 가능한 사용 방법 FTP目录 출력 이름입니다.
  45. // CURLOPT_HEADER=>true,// 사용하기 위한 용도는 다음과 같습니다.
  46. // CURLINFO_HEADER_OUT=>false, //求字符串。
  47. // CURLOPT_HTTPGET=>true,// GET, 因为GET是默认是, 所以只被修改情况下使用.
  48. // 엘 => 사실 ,// 启用时会HTTP代理来传输.
  49. // CURLOPT_MUTE=>true,// 启用时将cURL函数中所有修改过的参数恢复默认值。
  50. // OPT_NETRC=>거짓 ,// 에서 连接建立以后,访问~/.netrc文件获取用户name and 密码信息连接远程站点。
  51. // CURLOPT_NOBODY=>true, 신체 부분이 분리되었습니다.
  52. / / CURLOPT_NOPROGRESS=>false,//사용하는 컬이 사용하기에 적합합니다.
  53. // CURLOPT_NOSIGNAL=>false,//传递给php의 여행信号。이 항목은 SAPI 멀티스레드 전송 중에 기본적으로 활성화됩니다. cURL 7.10에 추가되었습니다.
  54. // CURLOPT_POST=>false,// 활성화되면 양식 제출과 마찬가지로 일반 POST 요청이 전송됩니다(application/x-www-form-urlencoded 입력).
  55. // CURLOPT_PUT=>false,// 활성화되면 HTTP가 파일을 보낼 수 있도록 허용합니다. CURLOPT_INFILE 및 CURLOPT_INFILESIZE를 동시에 설정해야 합니다.
  56. // CURLOPT_TRANSFERTEXT=>false,// 활성화되면 FTP 전송에 ASCII 모드를 사용합니다. LDAP의 경우 HTML이 아닌 일반 텍스트 정보를 검색합니다. Windows 시스템에서는 시스템이 STDOUT을 바이너리 모드로 설정하지 않습니다.
  57. // CURLOPT_UNRESTRICTED_AUTH=>true,// 도메인 이름이 변경된 경우에도 CURLOPT_FOLLOWLOCATION을 사용하여 생성된 헤더의 여러 위치에 사용자 이름 및 비밀번호 정보를 지속적으로 추가합니다.
  58. // CURLOPT_UPLOAD=>false,// 활성화되면 파일 업로드를 허용합니다.
  59. // CURLOPT_VERBOSE =>true,// 활성화되면 모든 정보가 보고되고 STDERR 또는 지정된 CURLOPT_STDERR에 저장됩니다.
  60. );
  61. private $headers = array();
  62. private $requests = array();
  63. private $requestMap = array();
  64. /*********************
  65. 콜백 함수 구성
  66. ********************/
  67. 함수 __construct($callback = null) {
  68. $this->callback = $callback;
  69. }
  70. /**************************************************** * ****************
  71. __get 메소드 오버로드
  72. *********************** * **********************************************/
  73. 공용 함수 __get($ name) {
  74. return (isset($this->{$name})) ? $this->{$name} : null;
  75. }
  76. /**************************************************** * *********************
  77. __set 메소드 오버로드
  78. ****************** *** **********************************/
  79. 공개 함수 __set($name, $value) {
  80. // 增加一个设置到headers
  81. if ($name == "options" || $name == "headers") {
  82. $this->{$name} = $value $this->{$name };
  83. } else {
  84. $this->{$name} = $value;
  85. }
  86. return true;
  87. }
  88. //增加一个请求
  89. 공개 function add($request) {
  90. $this->requests[] = $request;
  91. return true;
  92. }
  93. 공개 함수 요청($url, $method = "GET", $post_data = null, $headers = null, $options = null) {
  94. $this->requests[] = new request_setting($url, $method, $post_data, $headers, $options );
  95. true 반환;
  96. }
  97. 공개 함수 get($url, $headers = null, $options = null) {
  98. return $this->request($url, "GET", null, $headers, $options);
  99. }
  100. 공개 함수 post($url, $post_data = null, $headers = null, $options = null) {
  101. 반환 $this->request($url, "POST", $post_data, $headers, $options);
  102. }
  103. 비공개 함수 Single_curl() {
  104. $ch = cur_init(); //初始化
  105. $request = array_shift($this->requests);//把第一个单元移出并作为结果
  106. $options = $this->get_options($request);//获得该单代设置
  107. 컬_setopt_array($ch, $options);//批设置
  108. $output = 컬_exec($ch);
  109. $curl_info = 컬_getinfo($ch);
  110. if ($this->callback) {
  111. $callback = $this->callback;
  112. if (is_callable($this->callback)) {
  113. call_user_func($callback, $output, $curl_info, $request);
  114. }
  115. }
  116. else
  117. return $output;
  118. return true;
  119. }
  120. 비공개 함수 Rolling_curl($ thread_size = null) {
  121. if ($thread_size){
  122. $this->thread_size = $thread_size;
  123. }
  124. if (count($this->requests) < $this->thread_size){
  125. $this->thread_size = count($this->requests);
  126. }
  127. if ($this->thread_size < 2) {
  128. $errorinfo = '线程大小必须大于 1!!!!';
  129. throw new Exception($errorinfo);
  130. }
  131. $queue =curl_multi_init();
  132. //재线程里开始增加任务队列
  133. for ($i = 0; $i < $this->thread_size; $i ) {
  134. $ch = 컬_init();
  135. $options = $this->get_options($this->requests[$i]);
  136. 컬_setopt_array($ch, $options) ;//获得设置
  137. cur_multi_add_handle($queue, $ch);//添加进去
  138. $key = (string) $ch;
  139. $this->requestMap[$key] = $i;
  140. }
  141. do {
  142. while (($statu_run_muti_exec = cur_multi_exec($queue, $active)) == CURLM_CALL_MULTI_PERFORM) ;
  143. if ($statu_run_muti_exec != CURLM_OK){ break; }
  144. // 发现完成的一个请求,进行处理
  145. while ($done =curl_multi_info_read($queue)) {
  146. $curl_info =curl_getinfo($done['handle']);
  147. $ 출력 = 컬_멀티_getcontent($done['handle']);
  148. $callback = $this->callback;
  149. if (is_callable($callback)){
  150. $key = (문자열) $done[ 'handle'];
  151. $request = $this->requests[$this->requestMap[$key]];
  152. unset($this->requestMap[$key]);//这个销毁变一个未处理的请求加入到一个已经完成成队列中
  153. if ($i < count($this->requests) && isset($this->requests[$i]) && $i < count($this->requests)) {
  154. $ch =curl_init();
  155. $options = $this->get_options($this->requests[$i]);
  156. curl_setopt_array($ch, $options);
  157. curl_multi_add_handle($queue, $ch);
  158. $key = (문자열) $ch;
  159. $this->requestMap[$key] = $i;
  160. $i ;
  161. }
  162. cur_multi_remove_handle($queue, $done['handle']);
  163. echo "done ";
  164. print_r($queue);
  165. print_r ($done);
  166. }
  167. // 이 단계는 매우 중요합니다. 요청 중 하나가 완료되면 시간 초과 시간을 재설정해야 합니다.
  168. // 여기서 중요한 점은 모든 스레드에서 최소한 하나의 요청이 첫 번째 요청에 유효한지 확인하는 것입니다. 그렇지 않으면 처음에는 모든 것이 적용되지 않아 $active=0이 되므로 다음은 실행되지 않습니다.
  169. if ($active >0){
  170. cur_multi_select($queue, $this ->timeout );
  171. }
  172. } while ($active);
  173. curl_multi_close($queue);
  174. return true;
  175. }
  176. 공개 함수 실행($ thread_size = null) {
  177. //thread_size의 크기를 판단합니다. 요청이 하나만 있는 경우 단일 스레드 모드를 사용합니다
  178. if (count($this->requests) == 1) {
  179. return $this->single_curl( );
  180. } else {
  181. return $this->rolling_curl($thread_size);
  182. }
  183. }
  184. 비공개 함수 get_options($request) {
  185. $options = $this ->__get('options');
  186. if (ini_get('safe_mode') == 'Off' || !ini_get('safe_mode')) {
  187. // $options[CURLOPT_FOLLOWLOCATION] = 1;
  188. // $options[CURLOPT_MAXREDIRS] = 5;
  189. }
  190. $headers = $this->__get('headers');
  191. if ($request->options) {
  192. $options = $request->options $options;
  193. }
  194. $options[CURLOPT_URL] = $request->url;
  195. //The 다음은 각각 게시 옵션과 헤더 옵션입니다. ;
  196. }
  197. if ( $headers) {
  198. $options[CURLOPT_HEADER] = 0;
  199. $options[CURLOPT_HTTPHEADER] = $headers;
  200. }
  201. return $options;
  202. }
  203. 공개 함수 __destruct() {
  204. unset($this->thread_size, $this->callback, $this->options, $this->headers, $this- >요청);
  205. }
  206. }
  207. ?>
  208. 코드 복사
  1. header("content-type:text/html; charset=utf-8");
  2. require("muti_curl_class.php");set_time_limit(0);
  3. $sucesesnum=0;
  4. $good_proxy=array();
  5. function request_callback($response, $info, $request) {
  6. global $ ;
  7. // 下面的正規可以選擇性地顯示回傳的結果
  8. /* if (preg_match("~(.*?)~i", $response, $out )) {
  9. $title = $out[1];
  10. }*/
  11. // echo '
    '.$response .'
    ';
  12. echo '
    ';
  13. //對回應也就是 $response 進行偵測判斷裡面是否有設定的字符,如果有判斷運用該代理成功
  14. if( $response !== false && substr_count($response, 'User -agent: Baiduspider') >=1 ) {
  15. // $result = true;
  16. echo "true
    ";
  17. // echo $request[options][10004];
  18. // print_r ($request->options);
  19. echo $request->options[CURLOPT_PROXY];
  20. $good_proxy[]=$request->options[CURLOPT_PROXY];
  21. }
  22. }
  23. echo '
    the-->'. $sucesesnum.'// print_r ($request);
  24. //echo $request-> url;
  25. $sucesesnum ;
  26. echo "
    ";
  27. }
  28. $params = array_merge($_GET, $_POST); //此處取得傳遞過來的代理ip的位址
  29. $result = $proxy_ip = trim($params['ip']);
  30. $timeout=intval(trim($params['timeout']));
  31. if($timeoutif($timeout>300){$timeout=300;}
  32. $thread_size=intval(trim($params['thread_size']));
  33. if($thread_sizeize if($thread_size>300){$thread_size =300;}
  34. if($proxy_ip == '') {
  35. echo '請輸入IP!!';
  36. return;
  37. }
  38. $replace_arr1 = array(' ', 'qq代理:', 'dn28.com', 'qqip', 'qq代理', 'qqqipip ', '代理ip:', 'ip:', '代理ip','"',"'",'\','/',' ');
  39. $result = str_replace($replace_arr1, array (''), $result);
  40. $result = str_replace(",", "n", $result);
  41. $resArr = explode("n", $result);
  42. foreach( $resArr as $k => $v) {
  43. $posProxy = getPos($v, '@');
  44. if($posProxy===false){
  45. if (!empty($v )){$proxyip_and_port = $v; }
  46. }else{
  47. $proxyip_and_port = substr($v, 0, $posProxy);
  48. }
  49. $newRes[] =trim($proxyip_and_port) ;
  50. }
  51. print_r($newRes);
  52. //die();
  53. $option_setting = array(
  54. CURLOPT_SSL_VERIFYPEER => 0,
  55. CURLOPT_SSL_VERIFYPEER => 0, 5,
  56. CURLOPT_TIMEOUT => 30,
  57. CURLOPT_HEADER=>false,
  58. CURLOPT_PROXY=>'',///這個地方設定代理的位置
  59. );$btime=time();
  60. $rc = new muti_curl("request_callback");
  61. $rc->timeout = $timeout ;
  62. $rc->thread_size = $thread_size;
  63. foreach ($newRes as $v) {
  64. $option_setting[CURLOPT_PROXY]=$v;
  65. $request = new request_ $setting($, $setting( method = "GET", $post_data = null,$header= null, $option_setting);
  66. $rc->add($request);
  67. }
  68. $rc->execute();
  69. $etime=time();
  70. $usedtime=$etime-$btime;
  71. echo 'all'. $sucesesnum.'use'. $usedtime;
  72. echo '
    ';
  73. $good_proxy= array_unique($good_proxy);
  74. $str='';
  75. foreach ($good_proxy as $v){
  76. $str.="'".trim($v)."' ,";
  77. }
  78. $str= str_replace ( ' ' , '' ,$str );
  79. $str = preg_replace('/s /', ' ', $str);
  80. echo $str.'
    ';
  81. var_export ($good_proxy);
  82. //var_dump ($good_proxy);
  83. //*************************************** ************************************************** *********************
  84. //************************ *******只用了一個函數
  85. function parseProxyInfo ( $proxyStr ) {
  86. //$proxyStr = '202.115.207.25:80@HTTP;四川省成都市四川師範大學' ;
  87. $posIp = getPos($proxyStr, ':');
  88. $ip = substr($proxyStr, 0, $posIp);
  89. $posPort = getPos($proxyStr, '@');
  90. $port = substr($proxyStr, $posIp 1, $posPort-$posIp-1);
  91. $posType = getPos($proxyStr, ';');
  92. $type = substr($proxyStr , $posPort 1, $posType-$posPort-1);
  93. $location = substr(strstr($proxyStr, ';'), 1);
  94. return array(
  95. 'ip' => $ ip,
  96. 'port' => $port,
  97. 'type' => $type,
  98. 'location' => $location
  99. );
  100. }
  101. function getPos($haystack, $needle){ return strpos($haystack, $needle);}
  102. 函數check_proxy_is_useful($model, $proxy_info_arr = array()) {
  103. 全域$params, $config;
  104. if($model == 'single') {
  105. $proxy_port = intval(trim($params['port']));
  106. $check_proxy_url = $config['verify_url'];
  107. $proxy_time_out = intval(trim($params['timeout']));
  108. $retry = intval(trim($params['retry']));
  109. $proxy_ip = trim($params['ip']);
  110. $proxy = 新代理( $proxy_ip, $proxy_port, $check_proxy_url, $proxy_time_out, $retry );
  111. // 成功返回string success,失敗返回boolean false
  112. $result = $proxy -> check_proxy();
  113. //var_dump($result);
  114. $proxy_str_success = ''.$proxy_ip.':'.$proxy_port.'@'.'HTTP代理驗證成功! ';
  115. $proxy_str_failed = ''.$proxy_ip.':'.$proxy_port.'@'.'HTTP代理驗證失敗! ';
  116. 回傳$result !== false ? $proxy_str_success : $proxy_str_failed;
  117. } elseif ($model == 'collect') {
  118. $proxy_port = intval(triminfo_ ['port']));
  119. $check_proxy_url = $config['verify_url'];
  120. $proxy_time_out = intval(trim($params['timeout']));
  121. $retry = intval($params['timeout']));
  122. $retry = intval( trim($params['retry']));
  123. $proxy_ip = trim($proxy_info_arr[ 'ip']);
  124. /*echo $proxy_ip.'
    ';
  125. echo $ proxy_port.'
    ';
  126. echo $check_proxy_url.'';
  127. echo $proxy_time_out.'
    ';
  128. echo $retry.'
    ' ;*/
  129. if(!isset($proxy) )) {
  130. $proxy = new proxy( $proxy_ip, $proxy_port, $check_proxy_url, $proxy_time_out, $retry );
  131. }
  132. / / 成功回傳string success,失敗回傳boolean false
  133. $結果= $proxy -> check_proxy();
  134. return $result;
  135. }
  136. }
  137. function getget_single(){
  138. 全域$params, $config;
  139. $proxy_ip = trim($params ['ip']);
  140. if($proxy_ip == '') {
  141. echo '請輸入IP!! ';
  142. return;
  143. }
  144. echo check_proxy_is_useful('single' );
  145. }
  146. function get_proxy_by_collect(){
  147. function get_proxy_by_collect(){
  148. 38 美元, $config; $params['url'] = trim($params['url']);
  149. if($params['url'] == '') {
  150. echo '請輸入url!';
  151. return;
  152. }
  153. //$url = 'http:// www.dn28.com/html/75/n-5175.html';
  154. $con = iconv('GBK', ' UTF-8', file_get_contents($params['url']));
  155. preg_match ('/
(.*)
/s' , $con, $arr);
  • $result = strip_tags ($arr[1], '
    ');
  • $replace_arr1 = array(' ', 'QQ代理:', 'dn28.com ', 'qqip', 'QQ代理', ' qq代理ip', '代理ip:', 'ip:', '代理ip');
  • $result = str_replace($replace_arr1, array('') , $result);
  • //print_r( $arr);
  • $resArr =explode('
    ', $result); //print_r($resArr); echo '

    代理開始批量驗證中,整個整個過程將花費您幾十年的時間。

    ';
  • unset($_SESSION['success_arr']);
  • foreach($resArr as $k => $v) {
  • $newRes[$k] = parseProxyInfo( $v );
  • //print_r($newRes[$k]);
  • /*return;*/
  • $result = check_proxy_is_useful('collect', $newRes[$k]);
  • $ $ $ proxy_str_success = ''.$newRes[$k]['ip'].':'.$newRes[$k]['port'].'@'.$newRes[$k ]['型別']。 >'.$newRes[$k]['ip'].':'.$newRes[$k]['連接埠'].'@'.$newRes[$k]['類型'].' 代理驗證失敗! > $_SESSION['success_arr'][] = $success_arr[] = $newRes[$k];
  • } else {
  • echo $proxy_str_failed;
  • }
  • echo '
    ';
  • }
  • if(isset($success_arr) && count($success_arr) > 0 ) {
  • save_success_proxy($success_proxy($a) ; > echo '

    [將驗證成功的代理人儲存到本機]   [我要查看歷史資料]

    ';
  • } else {
  • echo '

    [我要看歷史資料]

    ';
  • }
  • //print_r($success_arr);
  • }
  • function get_proxy_by_rule(){
  • 全域$params, $config;
  • $result = $proxy_ip = trim($params['ip']);
  • if($proxy_ip == '') {
  • echo '請輸入IP!!';
  • return;
  • }
  • $replace_arr1 = array(' ', 'QQ代理:', 'dn28.com', 'qqip', 'QQ代理', 'QQ代理ip', '代理ip:', 'ip:', '代理ip');
  • $result = str_replace($replace_arr1, array(''), $ result);
  • $resArr =explode("n", $result);
  • //print_r( $resArr);
  • echo '

    代理開始批次驗證中,整個過程需要花費

    ';
  • unset($_SESSION['success_arr']);
  • foreach($resArr as $k => $v) {
  • $newRes[$k] = parseProxyInfo($v);
  • //print_r($newRes[$k]);
  • /*return;*/
  • $result = check_proxy_is_useful('collect', $newRes[$k]);
  • //var_dump($result);
  • $proxy_str_success = ''.$newRes[$ k] ['ip'].':'.$newRes[$k]['連接埠'].'@'.$newRes[$k]['類型'].' 代理驗證成功! ]['ip'].':'.$newRes[$k]['連接埠'].'@'.$newRes[$k]['類型'].' 代理驗證失敗! > $_SESSION['success_arr'][] = $success_arr[] = $newRes[$k];
  • } else {
  • echo $proxy_str_failed;
  • }
  • echo '
    ';
  • }
  • if(isset($success_arr) && count($success_arr) > 0 ) {
  • save_success_proxy($success_proxy($a) ; > echo '

    [儲存到php格式檔案]   [保存驗證成功的代理到本地電腦] ;  [我要看歷史資料]

    ';
  • } else {
  • echo '

    [我不在乎查看資料歷史]

    ';
  • }
  • }
  • function save_success_proxy($success_arr){
  • global $config;
  • date_default_timezone_set(' PRC');
  • date_default_timezone_set(' PRC'); $v) {
  • $str .= $v['ip'].':'.$ v['連接埠'].'@'.$v['型別' ].';'.$v['位置']."n";
  • }
  • $fp = fopen($config[ 'root_path'].'/success_proxy/'.date('YmdHi') .'.log', 'a ');
  • fwrite($fp, $str);
  • fclose($fp);
  • 取消設定($str);
  • }
  • ?>
  • 複製程式碼
  • 성명:
    본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.