编程新手看过来,一文教会你如何用PHP实现高并发服务器?
haoteby 2025-02-03 12:53 14 浏览
一提到高并发,就没有办法绕开I/O复用,再具体到特定的平台linux, 就没办法绕开epoll. epoll为啥高效的原理就不讲了,感兴趣的同学可以自行搜索研究一下。
01
php怎么玩epoll?
首先得安装个libevent库,再装个event扩展或者libevent扩展就可以愉快地玩耍了.
有些人搞不清楚libevent库跟libevent扩展的区别,简单来说:
libevent库是C语言对epoll的封装,跟PHP毛关系都没有;libevent扩展就是PHP跟libevent库的沟通桥梁。实际上PHP的很多扩展就是干这个事的,有一些优秀的C语言库,PHP想直接拿来用,就通过PHP扩展的方式接入到PHP。
libevent扩展和event扩展随便选一个装,我个人更喜欢event扩展,因为更面向对象一点。
自己去http://pecl.php.net里面搜跟自己PHP版本对应的扩展,下好编译安装一下就OK了。
电脑装了多个版本的PHP编译的时候注意一下,phpize的版本要对应上,别搞错了,典型的五步曲:
1 phpize
2 ./configure
3 make
4 make install
5 php -m | grep event #看看装上了没
02
如何用PHP实现高并发服务器?
我们要实现的服务器,传输层是TCP协议,应用层协议太多太复杂,限于篇幅,会简单地以HTTP服务器举个例子,HTTP协议本身就很复杂,要实现起来细节上有很多考究,我们也不会完全实现HTTP协议。
首先,创建一个socket三步曲:
socket_create、socket_bind、socket_listen,为什么是这三步曲呢?
很简单,不管你传输层协议是啥,你下面的网络层协议你得选个版本吧,IPV4还是IPV6,传输层工作方式你得选一个吧,全双工、半双工还是单工,TCP还是UDP你也得选一个吧,socket_create就是这三个选项;
确定了网络层和传输层,你得告诉我监听哪个端口吧,这就对应了socket_bind;
然后你得开启监听,并指定一个客户端的队列长度吧,这就是socket_listen干的事。
创建完了,同步阻塞咱就不介绍了,一个进程同时最多hold处一个连接,多几个连接同时请求,就得等呗,超过了socket_listen指定的队列长度,就得返回504了。
多进程也一样,几个进程就有几个并发,进程又是昂贵资源,而且进程的上下文切换费时费力,导致整个系统效率低下。
没关系,咱有epoll,hold住万千请求不是梦,先实现一个Reactor。libevent库就是Reactor模式,直接调用函数就是在使用Reactor模式,所以无需纠结到底php怎么实现Reactor模式。
上面的代码很简单,简单解释一下概念,EventBase就是个容器,里面装的Event实例,这么一说,上面的代码就非常好懂了。然后一个Server.
Connection
先创建Socket的三步曲,设置成非阻塞模式。然后把socket加到Reactor中监听可读事件,可读的意思就是,缓冲区有数据了,才可读。
可读事件发生,说明有新连接来了,用stream_socket_accept接收新连接Conn,把Conn放到Reactor中监听可读事件,可读事件发生,说明客户端有数据发送过来了,循环读直到没数据,然后把Conn放到Reactor中监听可写事件,可写事件发生,说明客户端数据发送完了,把协议组装一下写入响应。
应用层如果是HTTP协议要注意一下Connection: keep-alive头,因为要复用连接,不要一写完就关闭连接。
撸完收工,用ab测一下并发,加-k参数复用连接,i5+8G,3W的并发没啥问题,当然我们这儿没有磁盘I/O,实际情况要从磁盘读取文件,读文件要通过linux的系统调用,而且有几次的文件拷贝操作,花销比较大,常用的解决思路是sendfile,零拷贝直接从一个FD到另一个FD,效率比较高,缺点就是PHP没有现成的已经实现sendfile的扩展,得自己动手,开发成本有点高。
ab测试PO图:
这就是PHP实现高并发服务器的思路了,只要是用EPOLL解决的,思路都一样,都是三步曲,放到Reactor下监听FD事件。
当然这个只是最简单的模型,还有很多可以改进的地方,比如说多进程,抄袭一下nginx,一个主进程+N个工作进程,多进程的目的还是想利用多核并行工作。
C语言实现也是这样,只是你可能不用libevent库,自己封装EPOLL,毕竟libevent库有点重,你也用不到libevent的很多东西;
当然了,C语言有一堆的数据结构以及定义在数据结构上的操作要写,没有GC,自己管理内存,还要有良好的设计,上多进程还得搞一搞IPC进程间通信的东西,开发难度比PHP要大地多,开发周期也很长,有兴趣的同学可以自己撸一个玩。
如果你对作者的课程感兴趣,蓝桥云课上线了该老师的《PHP Socket 编程基础入门》课程,如果你是蓝桥云课的刷题会员或者是学习会员,可免费学习该课程。
《PHP Socket 编程基础入门》难度为中级,如果你完全没有接触过php编程语言,那么不推荐你学习该课程。你可以先从蓝桥云课上学习《PHP 编程入门》。
相关推荐
- 如何随时清理浏览器缓存_清理浏览器缓存怎么弄
-
想随时清理浏览器缓存吗?Cookieformac版是Macos上一款浏览器缓存清理工具,所有的浏览器Cookie,本地存储数据,HTML5数据库,FlashCookie,Silverlight,...
- Luminati代理动态IP教程指南配置代理VMLogin中文版反指纹浏览器
-
介绍如何使用在VMLogin中文版设置Luminati代理。首先下载VMLogin中文版反指纹浏览器(https://cn.vmlogin.com)对于刚接触Luminati动态ip的朋友,是不是不懂...
- mac清除工具分享,解除您在安全方面的后顾之忧
-
想要永久的安全的处理掉重要数据,删除是之一,使用今天小编分享的mac清除工具,为您的操作再增一层“保护”,小伙伴慎用哟,一旦使用就不可以恢复咯,来吧一起看看吧~mac清除工具分享,解除您在安全方面的后...
- 取代cookie的网站追踪技术:”帆布指纹识别”
-
【前言】一般情况下,网站或者广告联盟都会非常想要一种技术方式可以在网络上精确定位到每一个个体,这样可以通过收集这些个体的数据,通过分析后更加精准的去推送广告(精准化营销)或其他有针对性的一些活动。Co...
- 辅助上网为啥会被抛弃 曲奇(Cookie)虽甜但有毒
-
近期有个小新闻,大概很多小伙伴都没有注意到,那就是谷歌Chrome浏览器要弃用Cookie了!说到Cookie功能,很多小伙伴大概觉得不怎么熟悉,有可能还不如前一段时间被弃用的Flash“出名”,但它...
- 浏览器指纹是什么?浏览器指纹包括哪些信息
-
本文关键词:浏览器指纹、指纹浏览器、浏览器指纹信息、指纹浏览器原理什么是浏览器指纹?浏览器指纹是指浏览器的各种信息,当我们访问其他网站时,即使是在匿名的模式下,这些信息也可以帮助网站识别我们的身份。...
- 那些通用清除软件不曾注意的秘密_清理不常用的应用软件
-
系统清理就像卫生检查前的大扫除,即使你使出吃奶的劲儿把一切可能的地方都打扫过,还会留下边边角角的遗漏。随着大家电脑安全意识的提高,越来越多的朋友开始关注自己的电脑安全,也知道安装360系列软件来"武装...
- 「网络安全宣传周」这些安全上网小知识你要知道!
-
小布说:互联网改变了人们的衣食住行,但与之伴生的网络安全威胁也不容忽视。近些年来,风靡全球的勒索病毒、时有发生的电信诈骗、防不胜防的个人信息泄露时时刻刻都威胁着我们的生活。9月18日-24日是第四届...
- TypeScript 终极初学者指南_typescript 进阶
-
在过去的几年里TypeScript变得越来越流行,现在许多工作都要求开发人员了解TypeScript...
- jQuery知识一览_jquery的认识和使用
-
一、概览jQuery官网:https://jquery.com/jQuery是一个高效、轻量并且功能丰富的js库。核心在于查询query。...
- 我的第一个Electron应用_electronmy
-
hello,好久不见,最近笔者花了几天时间入门Electron,然后做了一个非常简单的应用,本文就来给各位分享一下过程,Electron大佬请随意~笔者开源了一个Web思维导图,虽然借助showSav...
- HTML5 之拖放(Drag 和 Drop)_html拖放api
-
简介拖放是一种常见的特性,即抓取对象以后拖到另一个位置。在HTML5中,拖放是标准的一部分,任何元素都能够拖放。先点击一个小例子:在用户开始拖动<p>元素时执行JavaScrip...
- 如何用JavaScript判断输入值是数字还是字母?
-
在日常开发中,我们有时候需要判断用户输入的是数字还是字母。本文将介绍如何用JavaScript实现这一功能。检查输入值是否是数字或字母...
- 图形编辑器开发:快捷键的管理_图形编辑工具
-
大家好,我是前端西瓜哥。...
- 浏览器原生剪贴板:原来它能这样读取用户截图!
-
当我们使用GitHub时,会发现Ctrl+V就能直接读取用户剪贴板图片进行粘贴,那么它是如何工作的?安全性如何?...