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

php多线程  时间:2021-01-31  阅读:()

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

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

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

1. $hosts = array("host1 .sample.com", "host2 .sample.com", "host3.sampl e.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. ?>

它运行的很好但是在fs ockop en()分析完ho stnam e并且建立一个成功的连接或者延时$timeo ut秒之前扩充这段代码来管理大量服务器将耗费很长时间。

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

有两种方法可以实现 PHP5中可以使用新增的str eam_s ocket_clie nt()函数直接替换掉fso cketo pen() 。PHP5之前的版本你需要自己动手用sock ets扩展解决问题。下面是PH P5中的解决方法

1. $hosts = array("host1 .sample.com", "host2 .sample.com", "host3.sampl e.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_CONNECT| STREAM_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. ?>

我们用st ream_selec t()等待soc kets打开的连接事件。 strea m_sel ect()调用系统的selec t(2)函数来工作前面三个参数是你要使用的str eams的数组你可以对其读取写入和获取异常分别针对三个参数 。 stream_sel ect()可以通过设置$timeout 秒参数来等待事件发生-事件发生时相应的sockets数据将写入你传入的参数。

下面是PH P4. 1.0之后版本的实现如果你已经在编译PH P时包含了socke ts(ext/socke ts)支持你可以使用根上面类似的代码只是需要将上面的st reams/files ystem函数的功能用ext/socke ts函数实现。主要的不同在于我们用下面的函数代替stream_soc ket_c lient ()来建立连接

1. // This value is correct for Linux, other systems have other values

2. define( 'EINPROGRES S' , 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 = EINPROGRES S;

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. ?>

现在用so cket_selec t()替换掉st ream_selec t() 用sock et_re ad()替换掉fr ead() 用sock et_wr ite()替换掉fwrite() 用sock et_cl ose()替换掉fc lose()就可以执行脚本了

PHP5的先进之处在于你可以用s tream_sele ct()处理几乎所有的str eam。例如你可以通过inc lude STDIN用它接收键盘输入并保存进数组你还可以接收通过pr oc_op en()打开的管道中的数据。

无忧云:洛阳BGP云服务器低至38.4元/月起;雅安高防云服务器/高防物理机优惠

无忧云怎么样?无忧云,无忧云是一家成立于2017年的老牌商家旗下的服务器销售品牌,现由深圳市云上无忧网络科技有限公司运营,是正规持证IDC/ISP/IRCS商家,主要销售国内、中国香港、国外服务器产品,线路有腾讯云国外线路、自营香港CN2线路等,都是中国大陆直连线路,非常适合免备案建站业务需求和各种负载较高的项目,同时国内服务器也有多个BGP以及高防节点。一、无忧云官网点击此处进入无忧云官方网站二...

ATCLOUD.NET-OVH海外高防云主机,采用KVM架构,稳定安全且便宜好用,仅3刀起

官方网站:点击访问ATCLOUD.NET官网优惠码:目前提供Cloud VPS与Storage VPS两款产品的六折优惠活动(续费同价,截止至2021年5月31日)优惠码:UMMBPBR20Z活动方案:一、型号CPU内存磁盘流量优惠价格购买链接VPS-1GB0.5×2.6+GHz1GB20GB1TB$3立即购买VPS-2GB1×2.6+GHz2GB50GB2TB$6立即购买VPS-4GB2×2.6...

ZJI(月付450元),香港华为云线路服务器、E3服务器起

ZJI发布了9月份促销信息,针对香港华为云线路物理服务器华为一型提供立减300元优惠码,优惠后香港华为一型月付仅450元起。ZJI是原来Wordpress圈知名主机商家:维翔主机,成立于2011年,2018年9月更名为ZJI,提供中国香港、台湾、日本、美国独立服务器(自营/数据中心直营)租用及VDS、虚拟主机空间、域名注册等业务,商家所选数据中心均为国内访问质量高的机房和线路,比如香港阿里云、华为...

php多线程为你推荐
软银集团日本软银是什么呀?304和316不锈钢哪个好304与316用哪个好?闪迪和金士顿哪个好闪迪和金士顿哪个好朱祁钰和朱祁镇哪个好明英宗和明代宗是怎么回事?网页传奇哪个好玩哪个传奇类网页游戏即好玩又不花钱??无纺布和熔喷布口罩哪个好一次性口罩的熔喷布和KN95的熔喷布有什么不同?红茶和绿茶哪个好红茶和绿茶哪个更好?播放器哪个好什么播放器好用云盘哪个好哪个网盘好用 而且下载速度快 还免费美国国际东西方大学明尼苏达大学(是莫瑞斯分校)和美国东北大学 应该去哪一个 是这个方面的专家回答啊!有偏见性的不要说!
政务和公益机构域名注册管理中心 长沙服务器租用 谷歌域名邮箱 主机优惠码 主机 天猫双十一抢红包 500m空间 本网站在美国维护 100m空间 ftp免费空间 t云 台湾谷歌 能外链的相册 卡巴斯基是免费的吗 架设邮件服务器 太原联通测速 中国电信网络测速 电信网络测速器 腾讯网盘 域名和主机 更多