0

0

巧妙解决PHP无法实现多线程的问题

php中文网

php中文网

发布时间:2016-06-21 08:51:02

|

1067人浏览过

|

来源于php中文网

原创

有没有办法在PHP中实现多线程呢?假设你正在写一个基于多台服务器的PHP应用,理想的情况时同时向多台服务器发送请求,而不是一台接一台。可以实现吗?当有人想要实现并发功能时,他们通常会想到用fork或者spawn threads,但是当他们发现PHP不支持多线程的时候,大概会转换思路去用一些不够好的语言,比如Perl。

 

其实的是大多数情况下,你大可不必使用fork或者线程,并且你会得到比用fork或thread更好的性能。假设你要建立一个服务来检查正在运行的n台服务器,以确定他们还在正常运转。你可能会写下面这样的代码:


  1. 		$hosts = array("host1.sample.com", "host2.sample.com", "host3.sample.com");  
  2. $timeout = 15;  
  3. $status = array();  
  4. foreach ($hosts as $host) {   
  5.         $errno = 0;   
  6.         $errstr = "";   
  7.         $s = fsockopen($host, 80, $errno, $errstr, $timeout);   
  8.         if ($s) {    
  9.              $status[$host] = "Connectedn";    
  10.              fwrite($s, "HEAD / HTTP/1.0rnHost: $hostrnrn");    
  11.             do {     
  12.                 $data = fread($s, 8192);     
  13.                 if (strlen($data) == 0) {     
  14.                 break;     
  15.                 }     
  16.              $status[$host] .= $data;    
  17.          }   
  18.          while (true);    
  19.             fclose($s);   
  20.           }   
  21.          else {    
  22.               $status[$host] = "Connection failed: $errno $errstrn";   
  23.          }  
  24. }  
  25. print_r($status);  
  26. ?> 

它运行的很好,但是在fsockopen()分析完hostname并且建立一个成功的连接(或者延时$timeout秒)之前,扩充这段代码来管理大量服务器将耗费很长时间。

因此我们必须放弃这段代码;我们可以建立异步连接-不需要等待fsockopen返回连接状态。PHP仍然需要解析hostname(所以直接使用ip更加明智),不过将在打开一个连接之后立刻返回,继而我们就可以连接下一台服务器。

有两种方法可以实现;PHP5中可以使用新增的stream_socket_client()函数直接替换掉fsocketopen()。PHP5之前的版本,你需要自己动手,用sockets扩展解决问题。下面是PHP5中的解决方法:


  1. 		$hosts = array("host1.sample.com", "host2.sample.com", "host3.sample.com");  
  2. $timeout = 15;  
  3. $status = array();  
  4. $sockets = array();  
  5. /* Initiate connections to all the hosts simultaneously */  
  6. foreach ($hosts as $id => $host) {   
  7.         $s = stream_socket_client("$host:80", $errno, $errstr, $timeout,    
  8.              STREAM_CLIENT_ASYNC_CONNECTSTREAM_CLIENT_CONNECT);   
  9.         if ($s) {    
  10.             $sockets[$id] = $s;    
  11.             $status[$id] = "in progress";   
  12.         }   
  13.         else {  $status[$id] = "failed, $errno $errstr";   
  14.         }  
  15. }  
  16. /* Now, wait for the results to come back in */  
  17.  
  18. while (count($sockets)) {   
  19.       $read = $write = $sockets;   
  20. /* This is the magic function - explained below */   
  21.       $n = stream_select($read, $write, $e = null, $timeout);   
  22.       if ($n > 0) {    
  23.       /* readable sockets either have data for us, or are failed   * connection attempts */    
  24.           foreach ($read as $r) {        
  25.                    $id = array_search($r, $sockets);        
  26.                    $data = fread($r, 8192);        
  27.           if (strlen($data) == 0) {     
  28.                    if ($status[$id] == "in progress") {      
  29.                        $status[$id] = "failed to connect";     
  30.                    }     
  31.           fclose($r);     
  32.           unset($sockets[$id]);        
  33.            }   
  34.            else {     
  35.                  $status[$id] .= $data;        
  36.            }    
  37.         }    
  38. /* writeable sockets can accept an HTTP request */    
  39. foreach ($write as $w) {     
  40.          $id = array_search($w, $sockets);     
  41.          fwrite($w, "HEAD / HTTP/1.0rnHost: "      
  42.          . $hosts[$id] .  "rnrn");     
  43.          $status[$id] = "waiting for response";    
  44.          }   
  45. }   
  46. else {    
  47. /* timed out waiting; assume that all hosts associated   * with $sockets are faulty */    
  48. foreach ($sockets as $id => $s) {     
  49.          $status[$id] = "timed out "   
  50.          . $status[$id];    
  51.          }    
  52. break;   
  53.   }  
  54. }  
  55. foreach ($hosts as $id => $host) {   
  56.       echo "Host: $hostn"; echo "Status: "   
  57.       . $status[$id] . "nn";  
  58. }   
  59. ?> 

我们用stream_select()等待sockets打开的连接事件。stream_select()调用系统的select(2)函数来工 作:前面三个参数是你要使用的streams的数组;你可以对其读取,写入和获取异常(分别针对三个参数)。stream_select()可以通过设 置$timeout(秒)参数来等待事件发生-事件发生时,相应的sockets数据将写入你传入的参数。

下面是PHP4.1.0之后版本的实现,如果你已经在编译PHP时包含了sockets(ext/sockets)支持,你可以使用根上面类似的代 码,只是需要将上面的streams/filesystem函数的功能用ext/sockets函数实现。主要的不同在于我们用下面的函数代替 stream_socket_client()来建立连接:


  1. 		// This value is correct for Linux, other systems have other values  
  2. define('EINPROGRESS', 115);  
  3. function non_blocking_connect($host, $port, &$errno, &$errstr, $timeout) {   
  4.         $ip = gethostbyname($host);   
  5.         $s = socket_create(AF_INET, SOCK_STREAM, 0);   
  6.         if (socket_set_nonblock($s)) {    
  7.            $r = @socket_connect($s, $ip, $port);    
  8.            if ($r  socket_last_error() == EINPROGRESS) {     
  9.                   $errno = EINPROGRESS;     
  10.                   return $s;    
  11.                }   
  12.          }   
  13.         $errno = socket_last_error($s);   
  14.         $errstr = socket_strerror($errno);   
  15.         socket_close($s);   
  16.         return false;  
  17. }  
  18. ?> 

现在用socket_select()替换掉stream_select(),用socket_read()替换掉fread(),用socket_write()替换掉fwrite(),用socket_close()替换掉fclose()就可以执行脚本了! PHP5的先进之处在于,你可以用stream_select()处理几乎所有的stream。例如你可以通过include STDIN用它接收键盘输入并保存进数组,你还可以接收通过proc_open()打开的管道中的数据。



PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
包子漫画网页版入口与全集阅读指南_正版免费漫画快速访问方法
包子漫画网页版入口与全集阅读指南_正版免费漫画快速访问方法

本专题汇总了包子漫画官网和网页版入口,提供最新章节抢先看方法、正版免费阅读指南,以及稳定访问方式,帮助用户快速直达包子漫画页面,无广告畅享全集漫画内容。

22

2026.02.10

MC.JS网页版快速畅玩指南_MC.JS官网在线入口及免安装体验方法
MC.JS网页版快速畅玩指南_MC.JS官网在线入口及免安装体验方法

本专题汇总了MC.JS官网入口和网页版快速畅玩方法,提供免安装访问、不同版本(1.8.8、1.12.8)在线体验指南,以及正版网页端操作说明,帮助玩家轻松进入MC.JS世界,实现即时畅玩与高效体验。

16

2026.02.10

谷歌邮箱网页版登录与注册全指南_Gmail账号快速访问与安全操作教程
谷歌邮箱网页版登录与注册全指南_Gmail账号快速访问与安全操作教程

本专题汇总了谷歌邮箱网页版的最新登录入口和注册方法,详细提供官方账号快速访问方式、网页版操作教程及安全登录技巧,帮助用户轻松管理Gmail邮箱账户,实现高效、安全的邮箱使用体验。

12

2026.02.10

铁路12306订票与退改全攻略_高效购票与座位选取技巧
铁路12306订票与退改全攻略_高效购票与座位选取技巧

本专题全面汇总铁路12306订票、退票、改签及候补订单操作技巧,提供车厢座位分布参考、抢票攻略和高铁安检注意事项,帮助新手用户快速掌握高效购票与退改流程,提高出行效率和体验。

10

2026.02.10

TensorFlow2深度学习模型实战与优化
TensorFlow2深度学习模型实战与优化

本专题面向 AI 与数据科学开发者,系统讲解 TensorFlow 2 框架下深度学习模型的构建、训练、调优与部署。内容包括神经网络基础、卷积神经网络、循环神经网络、优化算法及模型性能提升技巧。通过实战项目演示,帮助开发者掌握从模型设计到上线的完整流程。

0

2026.02.10

Vue3组合式API与组件开发实战
Vue3组合式API与组件开发实战

本专题讲解 Vue 3 组合式 API 的核心概念与应用技巧,深入分析响应式系统、生命周期管理、组件设计与复用策略。通过完整项目案例,指导前端开发者实现高性能、结构清晰的 Vue 应用,提升开发效率与代码可维护性。

4

2026.02.10

Go语言微服务架构与gRPC实战
Go语言微服务架构与gRPC实战

本专题面向有 Go 基础的开发者,系统讲解微服务架构设计与 gRPC 的高效应用。内容涵盖服务拆分、RPC 通信、负载均衡、错误处理、服务注册与发现等关键技术。通过实战案例,帮助开发者搭建高性能、可扩展的 Go 微服务系统。

1

2026.02.10

React 18状态管理与Hooks高级实践
React 18状态管理与Hooks高级实践

本专题专注于 React 18 的高级开发技术,详细讲解 useState、useEffect、useReducer、useContext 等 Hooks 的使用技巧,以及 Redux、Zustand 等状态管理工具的集成与优化方法。通过真实案例,帮助前端开发者构建可维护、性能优良的现代 React 应用。

4

2026.02.10

Node.js后端开发与Express框架实践
Node.js后端开发与Express框架实践

本专题针对初中级 Node.js 开发者,系统讲解如何使用 Express 框架搭建高性能后端服务。内容包括路由设计、中间件开发、数据库集成、API 安全与异常处理,以及 RESTful API 的设计与优化。通过实际项目演示,帮助开发者快速掌握 Node.js 后端开发流程。

2

2026.02.10

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Excel 教程
Excel 教程

共162课时 | 16.5万人学习

Pandas 教程
Pandas 教程

共15课时 | 1万人学习

C# 教程
C# 教程

共94课时 | 9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号