再谈PHP未来之路

这门语言现在到底处于生命周期的哪个阶段?其定位到底是怎样的?诸如 PHP7、Swoole 的出现到底能给 PHP 带来怎样的变化?

当我们拿 PHP 和 java 进行比较的时候,我们往往就两门语言本身进行比较,如一个是弱类型一个是强类型,一个是数组打天下一个是各种数据结构,甚至连花括号是不是换行写都会被讨论一番。但它们真正的区别并非这些。

当我们谈论一门语言的时候,我们是在谈它的生态。

“生态”一词在百度百科上的解释是:“生态一词,现在通常是指生物的生活状态。指生物在一定的自然环境下生存和发展的状态,也指生物的生理特性和生活习性。生态(Eco-)一词源于古希腊字,意思是指家(house)或者我们的环境”。

生态具有如下特点:

  1. 生态是系统,由多个部分组成的完整体;
  2. 生态是开放系统;
  3. 生态具有动态平衡性;
  4. 维持其动态平衡的是源动力,源动力一旦消失,生态即消亡。例如地球生态系统的源动力是太阳能;一旦太阳消失,地球生态则不复存在(想想《流浪地球》);

一种生物的生存状态不取决于生物自身,而取决于环境,就如恐龙的灭绝并非恐龙自身退化了,而是环境改变了(或者说恐龙的进化赶不上环境的变化)。

一门语言的兴衰不取决于它自身,而取决于环境,具体来说是环境中源动力的强弱。

PHP 应 Web 而生,考查其兴衰得考查互联网的发展。

一般认为互联网大致经历了三个阶段:

  • 阶段一:Web1.0 时代,传统的内容网站,如企业官网、行业门户网站等,网站自身产生内容,用户仅查看内容;
  • 阶段二:Web2.0 时代,用户参与内容的创建,如论坛、博客。阶段一和阶段二都是内容为主,服务为辅(虽然内容的产生方式有所不同);
  • 阶段三:移动互联网时代,信息流、内容与服务并存;

以上三个阶段的演化中,用户参与度越来越高,交互方式越来越丰富,网站流量越来越大。

阶段一和阶段二是 PHP 的黄金时代,从阶段二开始悄悄发生变化,而到了阶段三,PHP 的黄金时代基本结束。

PHP 这门语言的特点是“简单、实用”,入行门槛极低,一个编程小白,一周入门,两天出个网站。一个典型的例子,在数据结构上,不像其他语言有 Array、List、Map、Set,PHP 一个 Array 搞定所有的情况。

PHP 的这种“简单”是通过牺牲性能为代价的。由于需要简单,不能有各种类型限制,PHP 必须是动态语言;由于需要简单,能封装则封装,一个 file_x_contents 搞定文件(甚至是网络)读写(该函数是一次性将文件全部加载到内存中,很多人开发不考虑其局限性而用在所有场景,导致内存溢出);由于 Array 承包了所有集合型数据结构,其底层需要做各种处理不说,业务层也无法自主选择更合适的数据结构做针对性的优化(虽然后来 SPL 提供了一些基本数据结构)。

PHP 的这种“简单”还牺牲了另一样东西:程序员的专业素质。PHP 程序员根本不需要去了解真正的 Array 和 List 有什么区别,也不需要去管数据流、缓冲区。从长期来看,这一点是致命的,它使得 PHP 生态中的重要一环很脆弱,很可能是导致 PHP 最终衰落的真正因素。

在 Web1.0 时代,一方面内容产生者是网站自身,另一方面人们只能通过桌面浏览器上网,这些因素使得这个阶段绝大部分公司根本不会遇到高并发等性能问题,而且业务的简单性使得单体应用足以应付一切,因而这个阶段 PHP 的缺陷根本不足为患。于是,PHP 的优势(简单上手、快速开发)让这门语言大行其道,什么 JSP、ASP,根本不是对手。那个时期,人们谈论 java、C# 时,基本是在谈 ERP,只有 PHP 才是 Web。

到了 2.0 时代,论坛、博客、SNS 的出现,使得用户创建内容成为可能。由于用户的积极参与,网站服务器流量相对于 1.0 时代有了突增,特别是 SNS 的信息流特性,使得服务器面临相当的挑战。不过由于人们仍然是通过 PC 浏览器上网,在一定程度上限制了使用频率。这个时期,一些大公司针对 PHP 的性能缺陷做了自己的改造,如新浪的各种 c 扩展(yaf、yar 等),facebook 的 HVVM。

在这两个黄金时代,PHP 世界涌现了大量的经典开源项目:WordPress、ecshop、Magento、Discuz、Thinkphp、Yii 等。

彻底结束掉 PHP 黄金时代的是移动互联网的到来。iphone 改变了世界,也改变了 PHP 的命运。

移动互联网时代,人们随时随地都能上网,而且几乎每人一部手机,这带来的直接效果就是 Web 使用需求出现了数量级的增长。另外,移动互联网时代的另一个特点是内容+服务的一体化,网站不再只是提供内容,还提供服务(如各种 O2O),因而在使用频率、交互体验上的需求都大大增强。

举个例子,在 1.0 时代,浏览器和服务器根本不需要建立长连接,2.0 时代,由于信息流的出现,要求有轮询机制,但由于当时无论是浏览器还是 PHP 都不支持长连接,人们想了各种奇淫技巧来实现轮询。移动互联网时代,浏览器端有了 WebSocket,悲剧的是 PHP 本身却不支持 WebSocket(由于 PHP 的运行机制是一次请求后进程就结束了,在语言核心层面无法提供 WebSocket 机制。要想在核心层面支持 WebSocket,必须改造 PHP 的整个运行机制,这几乎是不可能的)。

至此,一方面 PHP 的性能问题成了致命问题,另一方面 PHP 各种“方便”的机制(如由 php-fpm 代替 PHP 脚本自身的常驻进程)满足不了新的场景需求,反倒成了桎梏。

在移动互联、万物成网的大背景下,微服务应运而生。一般认为微服务本身并非新的概念,早期的 SOA 就有其身影。不过我们谈论一个概念本身到底新不新没有意义(就好比有人认为中国的勾三股四弦五的发现比希腊的毕达哥拉斯定理要早,于是认为该定理是中国人发现的;有人认为中国的阴阳学说含有二进制思想,便认为二进制是中国人发明的),重要的是一个概念何时形成了一套完整的体系,以及是如何来解决实际问题的。

微服务架构是相对单体架构来说的。我们先说说微服务的缺点:服务间调用关系复杂、难治理、问题排查复杂、分布式事务问题等。既然有这么多缺点,为啥微服务架构当今能大行其道?原因在于单体架构解决不了当今面临的问题:巨大而复杂的业务群、高并发、高可用的系统需求。

微服务给 PHP 带来什么呢?

当我们将单体架构拆解成一个个小的服务的时候,我们来考查一下编程语言的选择,看看 PHP 还是不是最佳选择:

  • 首先微服务要轻量化。
  • 其次服务要被多个业务端调用,其运行要足够快。
  • 另外当服务间通信非常频繁时,通信协议要保持高效,此时 HTTP 协议并非最佳,很多公司倾向于 RPC 协议。
  • 后端服务相对于前面的业务层来说,变动频率相对要低一些,因而可以适当地牺牲一些开发效率。
  • 要有较成熟的生态和框架支持(成熟的服务治理生态)。

从上面几点来看,PHP 并非最佳选择:

  • 传统的 PHP 架构是 nginx + php-fpm + PHP script,显然不够轻量,成百上千个服务都驮着这么厚厚的一层壳,显然存在资源浪费问题。
  • PHP 作为脚本语言,由于存在脚本解析消耗,运行速度上赶不上 java、C++ 等静态语言(不过在 PHP 引入 opcode cache 后情况得到了很大改善,而且对于 Web 来说大部分时候都是 I/O 密集型操作,语言本身的性能影响对于绝大部分的公司来说并非主要问题————不过一方面心理学研究表明人类的认知并非完全理性的,人们认为 PHP 比 java 性能差那就是差,不管实际差多少(这就好比我们认为大品牌的东西一定比小品牌的好一样,编程语言的世界也有品牌效应))。
  • PHP 核心没有提供现成的 RPC 方案,但可以通过扩展解决,这不是问题。问题是传统的 PHP 架构(nginx + fpm + script,一次请求完成后工作进程即结束)并不能很好地应用 RPC 通信的优势。
  • 在生态和框架上,Swoole 貌似是个不错的选择,不过 Swoole 的微服务生态目前尚不成熟。
  • 大部分的 PHP 程序员对服务化比较陌生(以及对性能、可靠性等非功能性需求的普遍漠视),上手较慢。

综合考虑,大部分公司进行服务化的时候,会选用主流静态语言(java、C++ 以及后起之秀 golang 等)做服务,PHP 更多是来开发中间的业务聚合系统来调用这些服务。

至此,PHP 走下“神坛”,官方那句“PHP 是有史以来最好的语言”永成过去式。

不少人认为,PHP7 和 Swoole 给 PHP 在服务化时代带来新希望,因为理论上,上面提到的问题 PHP7 和 Swoole 都能较好的解决。

首先 PHP7 带来了极大的性能提升,而且引入强类型、严格模式等新特性,使得 PHP 越来越像强类型语言。其次 Swoole 的出现使得 PHP 很容易像 java、go 那样实现常驻进程服务而不需要依赖 nginx + php-fpm,那么 由“nginx + php-fpm + script” 的 CGI 模式在服务化时遇到的问题也都得到了很好的解决。

那么,PHP7(以及即将到来的 PHP8 的 JIT 特性)和 Swoole 能给 PHP 带来第二个黄金时代吗?

个人认为不能。还是那句话,当我们谈论语言时,实际上是在谈论生态。

编程语言的生态系统中有个很重要的角色:开发者群体。PHP 自出生时的目标就是“简单、强大、实用”,实现了高度的封装,让开发人员专心面对业务。这对工程是好事,对开发人员的成长(以及开发人员生态)来说却不是。绝大部分的 PHPer 都是业务工程师,几乎所有工作都是各种业务的 CRUD,很少涉及稍底层的东西,也鲜有关乎设计、架构的。在我周围的,以及面试遇到的,大部分人根本不了解设计模式、数据结构、算法、计算机原理,写出来的代码也仅仅是实现了业务的功能性需求,很少考虑非功能性需求。另外,在传统 PHP 的 CGI 模式下,PHP 脚本并不需要考虑自我恢复、自我保护能力如限流、重试、异步等这些在微服务架构下必须考虑的东西。

另外,由于大部分 PHP 程序员平时都是使用 MVC 框架提供的功能实现 CRUD,较少进行对象建模(PHP 并非生来就是面向对象语言,OO 特性是后面加进去的),导致大部分有相当工作经验的 PHPer 的建模能力都很弱,而微服务的一个重要工作就是对单体项目按业务领域进行拆分、建模,这对 PHPer 来说是个相当大的挑战。

一个结果是,PHP 程序员普遍专业素质都很弱,根本胜任不了复杂的系统架构————这里的复杂性有两个层面:技术层面和业务层面。

PHP7 和 Swoole 虽然弥补了语言自身的短板,却弥补不了生态中非语言部分的缺陷。有人认为这些缺陷是历史造成的,不能代表未来。万物的生命都是连续的、演化的,历史往往决定了未来,虽然身处现在的我们察觉不出。既然 PHP 生态在解决复杂系统问题时不具备优势,那么公司就会自然而然地选择其它更具优势的生态系统,自此便形成恶心循环(现实中我们遇到的情况是,很多使用 PHP 作为主要语言的中小公司业务规模上来后,不得不从外面聘请架构师,这些架构师大部分都是 java 出身,到公司第一件事就是强行 PHP 转 java)。

有人可能觉得我是 PHP 黑,毕竟我也没有做过严格的调查来得出上面的结论。但我们可以通过一些现象管中窥豹:

  • 我们可以很容易找到用 java、C++ 写的设计模式、数据结构与算法方面的畅销书,却几乎找不到 PHP 的。
  • 我们在博客园、CSDN 等技术博客上能看到大量 java、C++、C# 程序员的博客,却很少看到 PHP 的。
  • 我们看到技术博客上大量 java 程序员在谈论各种设计、服务、“三高”架构,却很少见到 PHP 的。
  • 我们能看到 java、C++ 程序员到处参加各种技术峰会,却很少见到 PHPer(除了 PHP 自己的专项会议)。

你会觉得仅凭 PHP7 与 Swoole 能让几乎不谈设计模式、不研究数据结构与算法、很少写博客、很少参加峰会的 PHPer 们开拓出一片服务化的新天地吗?

PHP 曾经辉煌过,在移动互联网之前,在单体为王的时代,就像 Delphi 在 Windows 桌面应用为王的时代取得的辉煌一样。现实的需求是语言生态系统的源动力,当需求发生不可逆转的改变时,午日终将西傍。

那么,接下来的问题是:PHP 会很快没落吗?

这个问题实际是在问:如今 PHP 是否还在某些场景下具有优势(即是否还存在现实需求这一源动力)?

PHP 的优势是简单、门槛低、实现功能快捷,很适合如下场景:

  • 业务、系统相对简单,无需服务化;
  • 对性能不是很敏感;
  • 需要快速实现、快速迭代;

在上面这些场景下,微服务(以及 java、C++ 等静态语言)的优点并不能弥补其缺点,因而推荐使用单体架构或者简单的服务化(仅仅进行主要服务拆分,并不引入复杂的服务治理体系),这种情形下 PHP 的优势就显现出来了。一般中小公司正是满足上面的场景,因而我们发现即使是在移动互联网时代 PHP 辉煌不再,但仍有大量中小公司采用 PHP 作为核心开发语言。

另外一个事实是,由于所有的大公司都是由小公司成长来的,在公司规模尚小的时候,他们大多也是采用 PHP 作为核心语言的,规模成长后,虽然 PHP 的各种短板阻碍了系统的发展,但由于已经有大量的 PHP 项目,完全重新用其他语言开发一遍不太现实,因而他们会采用各种优化手段,比如编写 PHP 扩展或者将 PHP 编译成某种静态语言(如 C++),或者将单体项目中的某些核心功能拆解成服务,单体项目调用后端服务接口————这种情况下,PHP 项目成了粘合层。

将 PHP 作为粘合语言的不光是因为历史遗留问题,还有不少公司新项目也会采用这种架构,这样既充分利用了 PHP 的开发效率(因为粘合层往往比较靠前端,需求变动较频繁,开发效率是必须要考虑的重要因素),也保证了核心服务的性能。

那么,接下来的问题是,作为快速原型语言和粘合层语言,有没有其他语言比 PHP 更具优势?

至少国内不用谈 Python 和 RoR(在国外这两者在 Web 开发上的占有率也不及 PHP),Python 程序员的重心已转大数据、人工智能了, RoR 至少在国内一直不温不火,在程序员的招聘上比 PHP 要难很多。

nodejs 曾经被认为是 PHP 的最大对手,一个很大的原因是人们认为如果一个公司使用 nodejs 作为后端语言,那么他只需要一样技术栈(前后端都是 js 程序员,而 js 程序员和 PHP 一样一抓一大把),体现了莫大的成本优势。但事实是 nodejs 并没有对 PHP 造成根本威胁,未来也不太可能会,原因是持上面观点的人认为统一技术栈就一定能节约成本,但这是个伪命题。一门语言具有解决某个问题的能力不代表人们就一定会拿它去解决问题,就好比 PHP 也能进行 socket 编程,但很少公司在生产环境大规模使用 PHP 编写服务器。js 天生就是 Web 前端语言,因而绝大部分 js 程序员都是一直做前端开发的,而前端开发和后端开发模式上有很大不同。前端在很长一段时间都是面向 DOM 编程,即使是有了模块化、React 这些新玩法后,前端开发的重心仍然是事件驱动的交互式编程。后端开发的重心在于建模(即使不对业务进行对象建模,也至少需要面向数据库进行数据建模)以及业务逻辑的实现,做后端开发,数据库、Linux 服务器是绕不开的,而这两者恰恰是大部分前端程序员所缺乏的(换句话说,要招一个既很熟悉前端开发又很熟悉后端开发的 js 程序员是非常难的)。结果就是,招一个 js 程序员用 nodejs 开发后端系统,其成本远大于招一个 PHPer。

因而,PHP 在未来可预见的很长时期内不会没落,它会作为中小公司的快速原型语言和大公司的粘合层语言长期存在。

另一个结论是:Python、Ruby On Rails、nodejs 这些语言虽然不会对 PHP 造成根本威胁,但会跟 PHP 一同在 Web 开发领域长期存在————因为它们的源动力是相同的,而 PHP 相对于它们的优势又不足以完全抹杀掉它们的存在。

总结:

最后,我将上面的分析总结成四个论断:

  • 论断一:PHP 在移动互联网到来之前出现过黄金时期,如今辉煌不再;
  • 论断二:PHP 在未来可预见的很长时期内不会没落;
  • 论断三:后黄金时代 PHP 的定位:中小公司的快速原型语言以及大公司的中间粘合层语言;
  • 论断四:PHP7 和 Swoole 让 PHP 在和其他同层级语言(如 Python、RoR、nodejs)的竞争中保持优势,但无法给 PHP 带来根本的变化(无法改变 PHP 的定位);

以上就是再谈PHP未来之路的详细内容,更多关于PHP的资料请关注我们其它相关文章!

(0)

相关推荐

  • 如何使用php生成zip压缩包

    压缩一个文件 我们将一个文件生成一个压缩包. <?php $path = "c:/wamp/www/log.txt"; $filename = "test.zip"; $zip = new ZipArchive(); $zip->open($filename,ZipArchive::CREATE); //打开压缩包 $zip->addFile($path,basename($path)); //向压缩包中添加文件 $zip->close();

  • PHP7下安装并使用xhprof性能分析工具

    该 xhprof 版本是从 https://github.com/longxinH/xhprof 获取 安装 xhprof cd xhprof/extension/ phpize ./configure  make 然后在/etc/php.ini中根据情况加入 extension=xhprof.so 执行 php -m | grep xhprof 可以看见输出,说明php扩展安装成功,然后重启Apache或者php-fpm 运行 可以直接运行从github上clone下来的文件里面example

  • 如何使用SublimeText3配置 PHP IDE环境

    首先是安装好PHP之后配置环境变量 然后在cmd中输入php -v 能看到版本号即为配置好了 之后在sublime中新建编译系统,输入代码 { "cmd": ["php", "$file"], "file_regex": "^(...*?):([0-9]*):?([0-9]*)", "selector": "source.php" } 然后保存在默认位置,改名字为ph

  • php中foreach遍历类对象的总结

    foreach 遍历数组很常见,同样foreach也可以遍历对象 做如下测试: class my { public $a = 'a'; protected $b = 'b'; private $c = 'c'; private $data = array('fantasy','windows','linux'); // 内部foreach遍历class function traversable() { foreach($this as $key=>$val) { echo $key.'=>';

  • 详解各种PHP函数漏洞

    1.MD5 compare漏洞 PHP在处理哈希字符串时,会利用"!="或"=="来对哈希值进行比较,它把每一个以"0E"开头的哈希值都解释为0,所以如果两个不同的密码经过哈希以后,其哈希值都是以"0E"开头的,那么PHP将会认为他们相同,都是0. 常见的payload有 0x01 md5(str) QNKCDZO 240610708 s878926199a s155964671a s214587387a s214587387

  • PHPStorm+Xdebug进行emote Debug时无法进入断点问题排查

    最近一直在研究Xdebug的用法,主要是因为Xdebug对PHP的开发效率提升确实很大,而却能帮我们解决很多问题.所以这里就总结下我在配置PHPStorm+Xdebug环境中遇到的坑,给遇到进不去断点的朋友一些排查的建议. 运行环境 PHP Version 7.1.22 Xdebug v2.6.0 PHP2016.2: 操作系统:MacOS 10.12.3 如果遇到无法进入断点的问题,可以按照以下的步骤进行排查 确认运行Web程序的PHP安装了Xdebug扩展 PHP永久了,难免会在电脑上多装几

  • php-fpm报502问题的解决办法

    搭建lnmp完lnmp环境后,测试时出现502报错,看到这个问题,我立刻想到是php-fpm没有起来,但是我用 ps -ef | grep php-fpm 截取 php-fpm 的进程,发现是有的,这时我有查看nginx的错误日志,发现错误信息是: 2017/05/05 17:08:45 [crit] 3258#0: *2 connect() to unix:/tmp/php-cgi.sock failed (2: No such file or directory) while connect

  • PHP远程调用以及RPC框架

    前言 一个项目,从开始到版本更新,一直到最后的版本维护.功能在不断增多,对应的代码量也在不断增加,也就意味着项目变得更不可维护,这时候,我们需要用拆分的方式将一个项目打散,以便开发团队更好的对项目进行维护. 分模块 这个阶段,一般也是项目的初级阶段,由于人手不够,一个服务端的接口项目只有一个开发进行维护,根据开发的习惯,会把项目分成若干个模块进行开发,在一个项目下进行部署. 这样做的缺点在于项目会随着版本更新而变得不可维护. 分项目 随着每个模块功能的不断完善,代码变得更加臃肿.这时候需要对项目

  • PHP代码加密和扩展解密实战

    这种方案是通过对代码进行加密,然后利用C语音写解密的PHP扩展.破解难度会有提升,但依然是会被破解的. 从网上找过各种代码加密的开源方案. 一旦开源,就不可能保证安全性.毕竟加密和解密的东西都是公开的. 目前我们没有能力自己去写扩展.还是需要采用开源的方案. 我找到的比较好用的是php-beast. https://github.com/liexusong/php-beast 实战开始 1.下载源码 wget https://github.com/liexusong/php-beast/arch

  • PHP实现创建以太坊钱包转账等功能

    1.Geth下载 (对以太坊了解不深的可以后面自己查找有关geth的资料,与此类似的客户端还有Ethereum.Parity .Mist),本文采用windows版(Windows系统:geth-windows-amd64-1.8.3) https://ethfans.org/wikis/Ethereum-Geth-Mirror 安装: windows版的直接傻瓜式安装到放软件的地方,然后打开安装根目录,这时可以看到有个geth.exe的可执行文件,先别急执行:先在该目录下创建一个piccgen

  • php常见的网络攻击及防御方法

    常见的Web攻击分为两类:一是利用Web服务器的漏洞进行攻击,如CGI缓冲区溢出,目录遍历漏洞利用等攻击;二是利用网页自身的安全漏洞进行攻击,如SQL注入,跨站脚本攻击等.下面这篇文章主要介绍了PHP安全防护之Web攻击,需要的朋友可以参考,下面来一起看看吧. SQL注入攻击(SQL Injection) 攻击者把SQL命令插入到Web表单的输入域或页面请求的字符串,欺骗服务器执行恶意的SQL命令.在某些表单中,用户输入的内容直接用来构造(或者影响)动态SQL命令,或作为存储过程的输入参数,这类

  • 详解PHP使用非对称加密算法RSA

    加密的类型 在日常设计及开发中,为确保数据传输和数据存储的安全,可通过特定的算法,将数据明文加密成复杂的密文.目前主流加密手段大致可分为单向加密和双向加密. 单向加密:通过对数据进行摘要计算生成密文,密文不可逆推还原.算法代表:Base64,MD5,SHA; 双向加密:与单向加密相反,可以把密文逆推还原成明文,双向加密又分为对称加密和非对称加密. 对称加密:指数据使用者必须拥有相同的密钥才可以进行加密解密,就像彼此约定的一串暗号.算法代表:DES,3DES,AES,IDEA,RC4,RC5; 非

随机推荐