PHP的运行机制与原理

Jackey PHP 3,435 次浏览 没有评论

PHP是什么

PHP是一种适用于Web开发的动态语言,是一个用C语言实现,包含大量组件的软件框架。

  • 多进程模型:PHP是多进程模型。不同请求间互不干涉,即一个请求挂掉不会对全盘服务造成影响。(使用进程控制函数,创建子进程,执行程序,处理信号等)。PHP也能支持多线程模型。

进程拥有一个完整的虚拟地址空间,不依赖于线程而独立存在;线程是进程的一部分,没有自己的地址空间,与进程内的其他线程一起共享分配给该进程的所有资源。
区别:
(1)进程具有独立的空间地址,一个进程崩溃后,在保护模式下不会对其它进程产生影响。
(2)线程只是一个进程的不同执行路径,线程有自己的堆栈和局部变量,但线程之间没有单独的地址空间,一个线程死掉就等于整个进程死掉。

  • 弱类型语言:和C、C++、C#、Java等不同,PHP是一种弱类型的语言。即一个变量的类型并不是一开始就确定不变的,运行中才会确定并可能发生隐式或显示的类型转换。
  • 引擎(Zend) + 组件(ext)的模式降低内部耦合
  • 中间层(sapi)隔绝web server和PHP
  • 语法简单灵活,没有太多规范

PHP的核心架构

从下向上如下:

  1. Zend引擎:Zend整体用C语言实现,是PHP的内核部分。它将PHP代码翻译,实现了基本的数据结构,内存分配机制及管理,提供了相应的api供外部调用,是一切的核心。
  2. Extensions:围绕Zend引擎,extensions通过组件式的方式提供各种基础服务,我们常见的各种内置函数、标准库等都是通过extension来实现,用户也可以根据需要实现自己的extension。
  3. Sapi:Aerver Application Programming Interface,即服务端应用编程接口,是PHP和web server的中间层。Sapi通过钩子函数,使PHP能和外部交互数据,这也将PHP和上层应用解耦。
  4. 上层应用:就是我们平时编写的PHP程序,通过不同的sapi方式得到各种各样的应用模式

Sapi

我们常见的Sapi有:

  1. apache2handler:以apache作为webserver,采用mod_PHP模式运行时候的处理方式,也是现在应用最广泛的一种。
  2. fast-cgi:这是webserver和PHP直接的另一种交互方式,也就是大名鼎鼎的fastcgi协议,在最近fastcgi+PHP得到越来越多的应用,也是异步webserver所唯一支持的方式。nginx就是通过php-fpm(fast-cgi)来解析php的。
  3. cli:命令行调用的应用模式

php+apache和php+nginx的区别

apache是通过mod_php来解析php的;nginx是通过php-fpm(fast-cgi)来解析php的。

  1. PHP解释器是否嵌入Web服务器进程内部执行
    mod_php通过嵌入PHP解释器到Apache进程中,只能与Apache配合使用。这种方式的弊端是内存占用大,如处理CSS、JS等静态文件完全没有必要加载解释器;而cgi和fast-cgi以独立的进程的形式出现,只要对应的Web服务器实现cgi或者fast-cgi协议,就能够处理PHP请求。
  2. 单个进程处理的请求数量
    mod_php和fast_cgi的模式在每个进程的生命周期内能够处理多个请求,而cgi 的模式处理一个请求就马上销毁进程,在高并发的场景下cgi 的性能非常糟糕。而fast-cgi也有启动多个进程,占用内存的缺点。

    cgi和fast-cgi是通信协议V1.0和V2.0,而php-cgi和php-fpm是实现了这种协议的程序。所以处理具体请求的是php-cgi或php-fpm,它们遵循fast-cgi协议。

综上,如果对性能有极高的要求,可以将静态请求和动态请求分开,这时Nginx+php-fpm是比较好的选择。

PHP的执行流程

拿到一段代码后,经过词法解析、语法解析等阶段后,源程序会被翻译成一个个指令(opcodes),然后ZEND虚拟机顺次执行这些指令完成操作。PHP本身是用C实现的,因此最终调用的也是C的函数,实际上,我们可以把PHP看做一个C开发的软件。

PHP变量

PHP是一门弱类型语言,本身不严格区分变量的类型。PHP变量可以分为简单类型(int、string、boolean)、集合类型(array、resource、object)和常量(const)。所有变量在底层都是同一种结构zval。
zval主要由三部分组成:

  • type:指定变量的类型
  • refcount&is_ref:用来实现引用计数
  • value:存储变量的实际数据(核心)。因为要存储多种类型,所以zvalue是一个union,也由此实现了弱类型。

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

Go