Zend framework 2 学习笔记

http://my.oschina.net/ohcoding/blog/145418#OSC_h2_1

Advertisements

zend framework resources

Zend Framework 2.0 (ZF2)官方资源

由于ZF2还在不断更新,比起从官方网站下载代码,更加推荐直接下载GIT库里的代码。

Zend Framework 2.0介绍及教程

首先可以通过下面的ppt对ZF2有一个全面的了解:

入门教程

ZF2的一些新概念

如何学习

ZF2由于本身的复杂性以及中文资源的稀缺,学习起来是有一定门槛。

所以建议有一定PHP基础的,对MVC有一定了解后再考虑使用ZF2框架。

ZF2大量使用了PHP的新特性,了解PHP5.3+有哪些新的特性和语法是必须的,比如php匿名函数(Closure)DateTime classLocale class等都是经常用的。

对ZF2的几个重要的新概念,包括DI、EventManager、ServiceManager等,同样建议有所了解,可以对开发中采用正确的模式和方法起到辅助作用。

如果遇到问题,Google关键词寻求答案应该是开发者的本能。如果实在解决不了,可以考虑在以下几个地方提出问题:

  • Stack Overflow 当然需要用英语提问,ZF2的作者也驻扎在上面,一个好问题也许可以得到非常好的解答。建议关注Stack Overflow的zend-framework2标签看看一些问题是如何解决的。
  • 德问SegmentFault 国内的编程问答社区,不过上面的Zend开发者都比较少。

笔者AlloVince在上述问答社区都有帐号,也欢迎邀请我作答。

Zend Framework 2 (ZF2)博客资源

Zend Framework 2 (ZF2) 实例程序

Zend Framework 2 (ZF2) 其他资源

php 执行原理 执行流程

简介
  先看看下面这个过程:

  • 我们从未手动开启过PHP的相关进程,它是随着Apache的启动而运行的;
  • PHP通过mod_php5.so模块和Apache相连(具体说来是SAPI,即服务器应用程序编程接口);
  • PHP总共有三个模块:内核、Zend引擎、以及扩展层;
  • PHP内核用来处理请求、文件流、错误处理等相关操作;
  • Zend引擎(ZE)用以将源文件转换成机器语言,然后在虚拟机上运行它;
  • 扩展层是一组函数、类库和流,PHP使用它们来执行一些特定的操作。比如,我们需要mysql扩展来连接MySQL数据库;
  • 当ZE执行程序时可能会需要连接若干扩展,这时ZE将控制权交给扩展,等处理完特定任务后再返还;
  • 最后,ZE将程序运行结果返回给PHP内核,它再将结果传送给SAPI层,最终输出到浏览器上。

深入探讨
  等等,没有这么简单。以上过程只是个简略版,让我们再深入挖掘一下,看看幕后还发生了些什么。

  • Apache启动后,PHP解释程序也随之启动;
  • PHP的启动过程有两步;
  • 第一步是初始化一些环境变量,这将在整个SAPI生命周期中发生作用;
  • 第二步是生成只针对当前请求的一些变量设置。

PHP启动第一步
  不清楚什么第一第二步是什么?别担心,我们接下来详细讨论一下。让我们先看看第一步,也是最主要的一步。要记住的是,第一步的操作在任何请求到达之前就发生了。

  • 启动Apache后,PHP解释程序也随之启动;
  • PHP调用各个扩展的MINIT方法,从而使这些扩展切换到可用状态。看看php.ini文件里打开了哪些扩展吧;
  • MINIT的意思是“模块初始化”。各个模块都定义了一组函数、类库等用以处理其他请求。

  一个典型的MINIT方法如下:
PHP_MINIT_FUNCTION(extension_name){
/* Initialize functions, classes etc */
}
PHP启动第二步

  • 当一个页面请求发生时,SAPI层将控制权交给PHP层。于是PHP设置了用于回复本次请求所需的环境变量。同时,它还建立一个变量表,用来存放执行过程中产生的变量名和值。
  • PHP调用各个模块的RINIT方法,即“请求初始化”。一个经典的例子是Session模块的RINIT,如果在php.ini中启用了Session模块,那在调用该模块的RINIT时就会初始化$_SESSION变量,并将相关内容读入;
  • RINIT方法可以看作是一个准备过程,在程序执行之间就会自动启动。

  一个典型的RINIT方法如下:
PHP_RINIT_FUNCTION(extension_name) {
/* Initialize session variables, pre-populate variables, redefine global variables etc */
}
PHP关闭第一步
  如同PHP启动一样,PHP的关闭也分两步:

  • 一旦页面执行完毕(无论是执行到了文件末尾还是用exit或die函数中止),PHP就会启动清理程序。它会按顺序调用各个模块的RSHUTDOWN方法。
  • RSHUTDOWN用以清除程序运行时产生的符号表,也就是对每个变量调用unset函数。

  一个典型的RSHUTDOWN方法如下:
PHP_RSHUTDOWN_FUNCTION(extension_name) {
/* Do memory management, unset all variables used in the last PHP call etc */
}
PHP关闭第二步
  最后,所有的请求都已处理完毕,SAPI也准备关闭了,PHP开始执行第二步:

  • PHP调用每个扩展的MSHUTDOWN方法,这是各个模块最后一次释放内存的机会。

  一个典型的RSHUTDOWN方法如下:
PHP_MSHUTDOWN_FUNCTION(extension_name) {
/* Free handlers and persistent memory etc */
}
  这样,整个PHP生命周期就结束了。要注意的是,只有在服务器没有请求的情况下才会执行“启动第一步”和“关闭第二步”。

下面的是用一些图示来说明的!

PHP底层工作原理

wps_clip_image-29471

图1 php结构

从图上可以看出,php从下到上是一个4层体系

①Zend引擎

Zend整体用纯c实现,是php的内核部分,它将php代码翻译(词法、语法解析等一系列编译过程)为可执行opcode的处理并实现相应的处理方法、实现了基本的数据结构(如hashtable、oo)、内存分配及管理、提供了相应的api方法供外部调用,是一切的核心,所有的外围功能均围绕zend实现。

②Extensions

围绕着zend引擎,extensions通过组件式的方式提供各种基础服务,我们常见的各种内置函数(如array系列)、标准库等都是通过extension来实现,用户也可以根据需要实现自己的extension以达到功能扩展、性能优化等目的(如贴吧正在使用的php中间层、富文本解析就是extension的典型应用)。

③Sapi

Sapi全称是Server Application Programming Interface,也就是服务端应用编程接口,sapi通过一系列钩子函数,使得php可以和外围交互数据,这是php非常优雅和成功的一个设计,通过sapi成功的将php本身和上层应用解耦隔离,php可以不再考虑如何针对不同应用进行兼容,而应用本身也可以针对自己的特点实现不同的处理方式。后面将在sapi章节中介绍

④上层应用

这就是我们平时编写的php程序,通过不同的sapi方式得到各种各样的应用模式,如通过webserver实现web应用、在命令行下以脚本方式运行等等。

构架思想:

引擎(Zend)+组件(ext)的模式降低内部耦合

中间层(sapi)隔绝web server和php

**************************************************************************

如果php是一辆车,那么

车的框架就是php本身

Zend是车的引擎(发动机)

Ext下面的各种组件就是车的轮子

Sapi可以看做是公路,车可以跑在不同类型的公路上

而一次php程序的执行就是汽车跑在公路上。

因此,我们需要:性能优异的引擎+合适的车轮+正确的跑道

Apache和php的关系

Apache对于php的解析,就是通过众多Module中的php Module来完成的。

wps_clip_image-31721

把php最终集成到Apache系统中,还需要对Apache进行一些必要的设置。这里,我们就以php的mod_php5 SAPI运行模式为例进行讲解,至于SAPI这个概念后面我们还会详细讲解。

假定我们安装的版本是Apache2 和 Php5,那么需要编辑Apache的主配置文件http.conf,在其中加入下面的几行内容:

Unix/Linux环境下:

LoadModule php5_module modules/mod_php5.so

AddType application/x-httpd-php .php

注:其中modules/mod_php5.so 是X系统环境下mod_php5.so文件的安装位置。

Windows环境下:

LoadModule php5_module d:/php/php5apache2.dll

AddType application/x-httpd-php .php

注:其中d:/php/php5apache2.dll 是在Windows环境下php5apache2.dll文件的安装位置。

这两项配置就是告诉Apache Server,以后收到的Url用户请求,凡是以php作为后缀,就需要调用php5_module模块(mod_php5.so/ php5apache2.dll)进行处理。

Apache的生命周期

wps_clip_image-8490

Apach的请求处理流程

wps_clip_image-17917

Apache请求处理循环详解
    Apache请求处理循环的11个阶段都做了哪些事情呢?

1、Post-Read-Request阶段

    在正常请求处理流程中,这是模块可以插入钩子的第一个阶段。对于那些想很早进入处理请求的模块来说,这个阶段可以被利用。

    2、URI Translation阶段
    Apache在本阶段的主要工作:将请求的URL映射到本地文件系统。模块可以在这阶段插入钩子,执行自己的映射逻辑。mod_alias就是利用这个阶段工作的。

    3、Header Parsing阶段
    Apache在本阶段的主要工作:检查请求的头部。由于模块可以在请求处理流程的任何一个点上执行检查请求头部的任务,因此这个钩子很少被使用。mod_setenvif就是利用这个阶段工作的。

    4、Access Control阶段
    Apache在本阶段的主要工作:根据配置文件检查是否允许访问请求的资源。Apache的标准逻辑实现了允许和拒绝指令。mod_authz_host就是利用这个阶段工作的。

    5、Authentication阶段
     Apache在本阶段的主要工作:按照配置文件设定的策略对用户进行认证,并设定用户名区域。模块可以在这阶段插入钩子,实现一个认证方法。

    6、Authorization阶段
    Apache在本阶段的主要工作:根据配置文件检查是否允许认证过的用户执行请求的操作。模块可以在这阶段插入钩子,实现一个用户权限管理的方法。

    7、MIME Type Checking阶段
    Apache在本阶段的主要工作:根据请求资源的MIME类型的相关规则,判定将要使用的内容处理函数。标准模块mod_negotiation和mod_mime实现了这个钩子。

    8、FixUp阶段
    这是一个通用的阶段,允许模块在内容生成器之前,运行任何必要的处理流程。和Post_Read_Request类似,这是一个能够捕获任何信息的钩子,也是最常使用的钩子。

    9、Response阶段
    Apache在本阶段的主要工作:生成返回客户端的内容,负责给客户端发送一个恰当的回复。这个阶段是整个处理流程的核心部分。

    10、Logging阶段
    Apache在本阶段的主要工作:在回复已经发送给客户端之后记录事务。模块可能修改或者替换Apache的标准日志记录。

11、CleanUp阶段
    Apache在本阶段的主要工作:清理本次请求事务处理完成之后遗留的环境,比如文件、目录的处理或者Socket的关闭等等,这是Apache一次请求处理的最后一个阶段。

LAMP架构:

wps_clip_image-24435

从下往上四层:

①liunx 属于操作系统的底层

②apache服务器,属于次服务器,沟通linux和PHP

③php:属于服务端编程语言,通过php_module 模块 和apache关联

    ④mysql和其他web服务:属于应用服务,通过PHP的Extensions外 挂模块和mysql关联

Android系统架构图

lamp和安卓的架构图比较一下,貌似和lamp架构有点相似,本人不懂安卓,只是感觉上有点相似,高手可以指出区别,小弟在此不胜感谢

wps_clip_image-27187

从上往下:

安卓架构————–说明——–LAMP架构

1.应用程序 ——–具体应用——–web应用

2.应用程序框架 —-java————-PHP语言和库

3.系统运行库 :—-虚拟机———WEB服务器

⒋Linux 内核 :—操作系统——-lamp架构中的L

深入了解php底层机制

作为一门动态语言,php是如何实现的,其底层机制如何,具有什么样的特点,本文深入浅出介绍了包括php设计理念、整体结构、核心数据结构和变量在内的相关底层知识,对我们更好的开发php程序,优化性能等有一定的指导意义。

TAG

Php 底层机制 性能优化

目录

1、概述… 1

what is php?. 1

了解它底层实现的目的?. 1

2、php的设计理念及特点… 1

3、Php的四层体系… 2

4、Sapi 3

5、Php的执行流程&opcode. 5

6、HashTable  — 核心数据结构… 6

7、Php变量… 8

概述… 8

Zval 9

整数、浮点数类型变量… 10

字符串变量… 10

数组变量… 11

资源类型变量… 11

Php变量的作用域… 12

1、概述

what is php?

一种适用于web开发的动态语言。具体点说:就是一个用c语言实现包含大量组件的软件框架。更狭义点看,可以把它认为是一个强大的ui框架。

了解它底层实现的目的?
  • 动态语言要像用好首先得了解它
  • 内存管理、框架模型值得我们借鉴
  • 通过扩展开发实现更多更强大的功能,优化我们程序的性能

2、php的设计理念及特点

  • 多进程模型

由于php是多进程模型,不同请求间互不干涉,这样保证了一个请求挂掉不会对全盘服务造成影响,当然,随着时代发展,php也早已支持多线程模型。

  • 弱类型语言

和c/c++、java、c#等语言不同,Php是一门弱类型语言:一个变量的类型并不是一开始就确定不变,运行中才会确定并可能发生隐式或显式的类型转换,这种机制的灵活性在web开发中非常方便、高效,具体会在后面php变量中详述。

  • 引擎(Zend)+组件(ext)的模式降低内部耦合
  • 中间层(sapi)隔绝web server和php
  • 语法简单灵活,没有太多规范。(导致风格混杂)
  • 再差的程序员也不会写出太离谱危害全局的程序。

3、Php的四层体系

Php的核心架构如下图

图1 php结构

从图上可以看出,php从下到上是一个4层体系

  • Zend引擎

Zend整体用纯c实现,是php的内核部分,它将php代码翻译(词法、语法解析等一系列编译过程)为可执行opcode的处理并实现相应的处理方法、实现了基本的数据结构(如hashtable、oo)、内存分配及管理、提供了相应的api方法供外部调用,是一切的核心,所有的外围功能均围绕zend实现。

  • Extensions

围绕着zend引擎,extensions通过组件式的方式提供各种基础服务,我们常见的各种内置函数(如array系列)、标准库等都是通过extension来实现,用户也可以根据需要实现自己的extension以达到功能扩展、性能优化等目的(如贴吧正在使用的php中间层、富文本解析就是extension的典型应用)。

  • Sapi

Sapi全称是Server Application Programming Interface,也就是服务端应用编程接口,sapi通过一系列钩子函数,使得php可以和外围交互数据,这是php非常优雅和成功的一个设计,通过sapi成功的将php本身和上层应用解耦隔离,php可以不再考虑如何针对不同应用进行兼容,而应用本身也可以针对自己的特点实现不同的处理方式。后面将在sapi章节中介绍

  • 上层应用

这就是我们平时编写的php程序,通过不同的sapi方式得到各种各样的应用模式,如通过webserver实现web应用、在命令行下以脚本方式运行等等。

如果php是一辆车,那么

  • 车的框架就是php本身
  • Zend是车的引擎(发动机)
  • Ext下面的各种组件就是车的轮子
  • Sapi可以看做是公路,车可以跑在不同类型的公路上

而一次php程序的执行就是汽车跑在公路上。

因此,我们需要:性能优异的引擎+合适的车轮+正确的跑道

4、Sapi

如前所述,sapi通过通过一系列的接口,使得外部应用可以和php交换数据并可以根据不同应用特点实现特定的处理方法,我们常见的一些sapi有:

  • apache2handler

这是以apache作为webserver,采用mod_php模式运行时候的处理方式,也是现在应用最广泛的一种。

  • cgi

这是webserver和php直接的另一种交互方式,也就是大名鼎鼎的fastcgi协议,在最近今年fastcgi+php得到越来越多的应用,也是异步webserver所唯一支持的方式。关于fastcgi和mod_php,可以参见另外一篇文章《php性能调研-mod_php vs fastcgi》

  • cli

命令行调用的应用模式

         Sapi的定义及主要接口函数如下图

图2 Sapi协议

这里介绍一下其中一些主要函数

  • startup:php被调用时初始化操作

比如cgi模式,在startup的时候会加载所有的extension并执行模块初始化工作。

  • shutdown:php关闭时收尾工作
  • activate:请求初始化
  • dectivate:请求结束时收尾工作
  • ub_write:指定数据输出方式

比如apache2handler方式,由于php作为apache的一个so存在,因此其输出也就是调用apache的ap_write函数,而在cgi模式下,会系统调用write。

  • sapi_error:错误处理函数
  • read_post:读取post数据
  • register_server_variables:往$_SERVER中注册环境变量

这个一般根据不同协议标准注册注册的变量。

5、Php的执行流程&opcode

我们先来看看php代码的执行所经过的流程。

图3 php代码的执行过程

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

         通过上面描述不难看出,php的执行的核心是翻译出来的一条一条指令,也即opcode

opcode

Opcode是php程序执行的最基本单位。一个opcode由两个参数(op1,op2)、返回值和处理函数组成。Php程序最终被翻译为一组opcode处理函数的顺序执行

常见的几个处理函数

         ZEND_ASSIGN_SPEC_CV_CV_HANDLER : 变量分配 ($a=$b)

         ZEND_DO_FCALL_BY_NAME_SPEC_HANDLER:函数调用

         ZEND_CONCAT_SPEC_CV_CV_HANDLER:字符串拼接 $a.$b

         ZEND_ADD_SPEC_CV_CONST_HANDLER: 加法运算 $a+2

         ZEND_IS_EQUAL_SPEC_CV_CONST:判断相等 $a==1

         ZEND_IS_IDENTICAL_SPEC_CV_CONST:判断相等 $a===1

6、HashTable  — 核心数据结构

HashTable是zend的核心数据结构,在php里面几乎并用来实现所有常见功能,我们知道的php数组即是其典型应用,此外,在zend内部,如函数符号表、全局变量等也都是基于hash table来实现。

         php的hash table具有如下特点:

  • 支持典型的key->value查询
  • 可以当做数组使用
  • 添加、删除节点是O(1)复杂度
  • key支持混合类型:同时存在关联数组合索引数组
  • Value支持混合类型:array (“string”,2332)
  • 支持线性遍历:如foreach

Zend hash table实现了典型的hash表散列结构,同时通过附加一个双向链表,提供了正向、反向遍历数组的功能。其结构如下图

图4 zend hash table数据结构

可以看到,在hash table中既有key->value形式的散列结构,也有双向链表模式,使得它能够非常方便的支持快速查找和线性遍历。

  • 散列结构

Zend的散列结构是典型的hash表模型,通过链表的方式来解决冲突。需要注意的是zend的hash table是一个自增长的数据结构,当hash表数目满了之后,其本身会动态以2倍的方式扩容并重新元素位置。初始大小均为8。

另外,在进行key->value快速查找时候,zend本身还做了一些优化,通过空间换时间的方式加快速度。比如在每个元素中都会用一个变量nKeyLength标识key的长度以作快速判定。

  • 双向链表

Zend hash table通过一个链表结构,实现了元素的线性遍历。理论上,做遍历使用单向链表就够了,之所以使用双向链表,主要目的是为了快速删除,避免遍历。

         Zend hash table是一种复合型的结构,作为数组使用时,即支持常见的关联数组也能够作为顺序索引数字来使用,甚至允许2者的混合。

  • Php关联数组

关联数组是典型的hash_table应用。一次查询过程经过如下几步

          getKeyHashValue h;

          index = n & nTableMask;

          Bucket *p = arBucket[index];

          while (p) {

              if ((p->h == h) && (p->nKeyLength == nKeyLength)) {

                 RETURN p->data;   

                    }

               p=p->next;

    }

RETURN FALTURE;

从代码可以看出,这是一个常见的hash查询过程并增加一些快速判定加速查找。

  • Php索引数组

索引数组就是我们常见的数组,通过下标访问。例如 $arr[0]

Zend HashTable内部进行了归一化处理,对于index类型key同样分配了hash值和nKeyLength(为0)。内部成员变量nNextFreeElement就是当前分配到的最大id,每次push后自动加一。

正是这种归一化处理,php才能够实现关联和非关联的混合

由于push操作的特殊性,索引key在php数组中先后顺序并不是通过下标大小来决定,而是由push的先后决定。

例如 $arr[1] = 2; $arr[2] = 3;

对于double类型的key,Zend HashTable会将他当做索引key处理

7、Php变量

概述
  • Php是一门弱类型语言,本身不严格区分变量的类型。
  • Php在变量申明的时候不需要指定类型。
  • Php在程序运行期间可能进行变量类型的隐示转换。
  • 和其他强类型语言一样,程序中也可以进行显示的类型转换。
  • Php变量可以分为简单类型(int、string、bool)、集合类型(array resource object)和常量(const)
  • 以上所有的变量在底层都是同一种结构 zval
Zval

Zval是zend中另一个非常重要的数据结构,用来标识并实现php变量,其数据结构如下

Zval主要由三部分组成:

1、  type:指定了变量所述的类型(整数、字符串、数组等)

2、  refcount&is_ref:用来实现引用计数(后面具体介绍)

3、  value:核心部分,存储了变量的实际数据

u  zvalue

Zvalue是用来保存一个变量的实际数据。因为要存储多种类型,所以zvalue是一个union,也由此实现了弱类型。

Php变量类型和其实际存储对应关系如下

IS_LONG   -> lvalue

IS_DOUBLE -> dvalue

IS_ARRAY  -> ht

IS_STRING -> str

IS_RESOURCE -> lvalue

u  引用计数

引用计数在内存回收、字符串操作等地方使用非常广泛。Php中的变量就是引用计数的典型应用

Zval的引用计数通过成员变量is_ref和ref_count实现,通过引用计数,多个变量可以共享同一份数据。避免频繁拷贝带来的大量消耗

在进行赋值操作时,zend将变量指向相同的zval同时ref_count++,在unset操作时,对应的ref_count-1。只有ref_count减为0时才会真正执行销毁操作

如果是引用赋值,则zend会修改is_ref为1

u  写时拷贝

Php变量通过引用计数实现变量共享数据,那如果改变其中一个变量值呢?

当试图写入一个变量时,Zend若发现该变量指向的zval被多个变量共享,则为其复制一份ref_count为1的zval,并递减原zval的refcount,这个过程称为“zval分离”。可见,只有在有写操作发生时zend才进行拷贝操作,因此也叫copy-on-write(写时拷贝)

对于引用型变量,其要求和非引用型相反,引用赋值的变量间必须是捆绑的,修改一个变量就修改了所有捆绑变量。

整数、浮点数类型变量

整数、浮点数是php中的基础类型之一,也是一个简单型变量。

对于整数和浮点数,在zvalue中直接存储对应的值。其类型分别是long和double。

从zvalue结构中可以看出,对于整数类型,和c等强类型语言不同,php是不区分int、unsigned int、long、long long等类型的,对它来说,整数只有一种类型也就是long。由此,可以看出,php里面,整数的取值范围是由编译器位数来决定而不是固定不变的

对于浮点数,类似整数,它也不区分float和double而是统一只有double一种类型。

  • 在php中,如果整数范围越界了怎么办?

这种情况下会自动转换为double类型,这个一定要小心,很多trick都是由此产生。

字符串变量

和整数一样,字符变量也是php中的基础类型和简单型变量

通过zvalue结构可以看出,在php中,字符串是由由指向实际数据的指针和长度结构体组成,这点和c++中的string比较类似。

由于通过一个实际变量表示长度,和c不同,它的字符串可以是2进制数据(包含),同时在php中,求字符串长度strlen是O(1)操作。

在新增、修改、追加字符串操作时,php都会重新分配内存生成新的字符串。

最后,出于安全考虑,php在生成一个字符串时末尾仍然会添加

  • 常见的字符串拼接方式及速度比较

假设有如下4个变量:

$strA=‘123’;  $strB = ‘456’; $intA=123; intB=456;

现在对如下的几种字符串拼接方式做一个比较和说明

1、$res = $strA.$strB和$res = “$strA$strB”

         这种情况下,zend会重新malloc一块内存并进行相应处理,其速度一般

2、$strA = $strA.$strB

         这种是速度最快的,zend会在当前strA基础上直接relloc,避免重复拷贝

3、$res = $intA.$intB

这种速度较慢,因为需要做隐式的格式转换,实际编写程序中也应该注意尽量避免

4、$strA = sprintf (“%s%s”,$strA.$strB);

这会是最慢的一种方式,因为sprintf在php中并不是一个语言结构,本身对于格式识别和处理就需要耗费比较多时间,另外本身机制也是malloc。不过sprintf的方式最具可读性,实际中可以根据具体情况灵活选择。

数组变量

如前所述,Php的数组通过Zend HashTable来天然实现

  • foreach操作如何实现?

         对一个数组的foreach就是通过遍历hashtable中的双向链表完成。对于索引数组,通过foreach遍历效率比for高很多,省去了key->value的查找

Count操作直接调用HashTable->NumOfElements,O(1)操作

对于’123’这样的字符串,zend会转换为其整数形式。$arr[‘123’]和$arr[123]是等价的

资源类型变量

         这是php中最复杂的一种变量,也是一种复合型结构。

Php的zval可以表示广泛的数据类型,但是对于自定义的数据类型却很难充分描述。由于没有有效的方式描绘这些复合结构,因此也没有办法对它们使用传统的操作符。要解决这个问题,只需要通过一个本质上任意的标识符(label)引用指针,这种方式被称为资源

在zval中,对于resource,lval作为指针来使用,直接指向资源所在的地址。

Resource可以是任意的复合结构,我们熟悉的mysqli、fsock、memcached等都是资源。

  • 使用资源
  • • 注册

对于一个自定义的数据类型,要想将它作为资源。首先需要进行注册,zend会为它分配全局唯一标示

  • • 获取一个资源变量

对于资源,zend维护了一个id->实际数据的hash_tale。对于一个resource,在zval中只记录了它的id。fetch的时候通过id在hash_table中找到具体的值返回

  • • 资源销毁

资源的数据类型是多种多样的。Zend本身没有办法销毁它。因此需要用户在注册资源的时候提供销毁函数。当unset资源时,zend调用相应的函数完成析构。同时从全局资源表中删除它。

  • 持久化资源

资源可以长期驻留,不只是在所有引用它的变量超出作用域之后,甚至是在一个请求结束了并且新的请求产生之后。这些资源称为持久资源,因为它们贯通SAPI的整个生命周期持续存在,除非特意销毁。

很多情况下,持久化资源可以在一定程度上提高性能。比如我们常见的mysql_pconnect ,持久化资源通过pemalloc分配内存,这样在请求结束的时候不会释放。

对zend来说,对两者本身并不区分。

Php变量的作用域
  • Php中的局部变量和全局变量是如何实现的?

         对于一个请求,任意时刻php都可以看到两个符号表(symbol_table和active_symbol_table),其中前者用来维护全局变量。后者是一个指针,指向当前活动的变量符号表,当程序进入到某个函数中时,zend就会为它分配一个符号表x同时将active_symbol_table指向a。通过这样的方式实现全局、局部变量的区分

  • 获取变量值

          php的符号表是通过hash_table实现的,对于每个变量都分配唯一标识,获取的时候根据标识从表中找到相应zval返回

  • 函数中使用全局变量

         在函数中,我们可以通过显式申明global来使用全局变量。在active_symbol_table中创建symbol_table中同名变量的引用,如果symbol_table中没有同名变量则会先创建。

在 Zend Developer Cloud上构建程序 – Zend Studio中文版

Zend应用程序部署向导 一文中,我们介绍了如何在Zend Studio中使用deployment.xml文件部署你的应用程序。由于Zend Studio,Zend Developer Cloud云支持以及GitHub项目三者实现完美集成。因此,本文将主要介绍如何在Zend Studio中部署,调试和修改你的GitHub项目。

》》下载和安装最新版Zend Studio 9.0

Step 1:创建Github和Zend Developer Cloud账户

在使用Github和Zend Developer Cloud前,需要创建有效账户。如果有账户,直接就可以进行Step 2的操作老。

1、创建GitHub账户

1)打开GitHub网站https://github.com/
2)点击欢迎页面的中心点击Plans, Pricing和Signup
3)点击 Create a free account。
4)输入账户信息,然后点击 Create an account。
5)根据GitHub 主页上的指示,创建一个项目和资源库。

2、创建Zend Developer Cloud帐户

1)打开 Zend Developer Cloud 网址:http://www.phpcloud.com/
2)在页面的右上方,点击 Login here。

在 Zend Developer Cloud上构建程序 - Zend Studio中文版用户指南(连载)
3)现在的Zend Developer Cloud平台针对的试用版。如果你已经有一个账户,你直接点击跳转到下一步。可。
4)点击 I do not have an account,创建一个新账户。
5)输入账户信息,然后点击Register。
6)回答页面上的验证问题,点击 Finish。
7)输入登录信息,然后点击Log In。
8)阅读条款,点击 I accept the terms, sign me up!完成注册。
9)点击Start now。

Step 2:开发GitHub项目

注册完GitHub和Zend Developer Cloud 账户后,你就可以使用嵌入在Zend Studio 和 Zend Developer Cloud 中的集成功能了。你可以在Zend Studio上,开发,测试和部署GitHub 程序
了。

在Zend Studio中创建新项目前,你需要在云中创建一个新的容器来保存应用程序。

1、进入Zend Developer Cloud homepage,打开你的账户,然后在欢迎页面上点击Start Now,此时,出现 Create Container 页面。

在 Zend Developer Cloud上构建程序 - Zend Studio中文版用户指南(连载)

2、在相应字段中输入容器的名称和密码,然后点击Create Container。出现SSH Keypair created窗口。

3、点击Download private key in PEM format。

Step 3:在Zend Studio中创建一个新GitHub项目

1、按照File | New | PHP Project from GitHub步骤操作后,出现Create Project from GitHub对话框。如果没有安装GitHub插件,Zend Studio 将自动开始安装该插件。详情见自定义和注册Zend Studio

在 Zend Developer Cloud上构建程序 - Zend Studio中文版用户指南(连载)

2、输入以下信息:

Project Name: helloworld
User: gadigold

3、点击Refresh。

4、点击Next,出现 Launch Settings 对话框。

在 Zend Developer Cloud上构建程序 - Zend Studio中文版用户指南(连载)

5、选择Zend Application Deployment,点击PHPCloud图标,创建一个新的项目。
出现Add Target 对话框。

在 Zend Developer Cloud上构建程序 - Zend Studio中文版用户指南(连载)

6、输入账户登陆信息,点击Test Connection。
7、如果没有问题出现,单击“ Finish”。
8、从下列列表中选择新的目标,然后单击Finish。

在 Zend Developer Cloud上构建程序 - Zend Studio中文版用户指南(连载)

Step 4: 在 Zend Developer Cloud上发布应用程序

1、创建项目后,打开deployment.xml文件

在 Zend Developer Cloud上构建程序 - Zend Studio中文版用户指南(连载)

2、在 Testing区内,点击Launch a PHP Application。

在 Zend Developer Cloud上构建程序 - Zend Studio中文版用户指南(连载)

3、当提示容器的真实性时,点击Finish 和 Yes 。

4、应用程序出现在Zend Developer Cloud网站的容器页面中,并且直接通过Zend Studio的内置浏览器打开。

Step 5:在Zend Studio中调试应用程序

1、在测试区中,点击 Launch a PHP Application in Debug Mode。

在 Zend Developer Cloud上构建程序 - Zend Studio中文版用户指南(连载)

2、继续点击调试按钮。

INSTALLING PHPUNIT FOR PHP 5.3 ON ZENDSERVER

ZendServer installs PHP CLI as part of the installation, and, as is customary for PHP 4.3+, includes a PEAR installer. It’s a good idea to install PEAR before installing PHPUnit as per the recommendation here.

  1. Open Zend\ZendServer\bin\go-pear.bat and change the line:
    %PHP_BIN% -d output_buffering=0 -d PEAR\go-pear.pharto (see PHAR Runtime Configuration):%PHP_BIN% -d output_buffering=0 -d phar.require_hash=0 PEAR\go-pear.pharThis will stop the fatal error:phar...does not have a signatureas by default PHP will not process a PHAR archive without a signature.
  2. Run go-pear.bat

    For Windows 7, make sure you open the console as an administrator, or the installer won’t be able to create some folders in the default install locations. At the command prompt enter:#cd \program files\zend\zendserver\bin
    # go-pear.bat
    and you should get something like this (I selected the system-wide install and accepted the default locations):install pear

    If the installation succeeded, you should see something like:end of pear install

    Note that c:\program files\zend\zendserver\bin is already in Windows Path, thanks to ZendServer.

  3. Update PEAR Trying to install PHPUnit at this point will fail, complaining of an outdated installer. Update the PEAR package:# pear channel-update pear.php.net
    # pear upgrade pear
  4. Add the PHPUnit channels – see Installing PHPUnit
    # pear channel-discover pear.phpunit.de
    # pear channel-discover pear.symfony-project.com
  5. Finally install PHPUnit# pear install phpunit/PHPUnitIf the installation runs successfully you should see:installing phpunit

You can check your PHPUnit by running it from the command prompt:# phpunit

INSTALLING PHPUNIT FOR PHP 5.3 ON ZENDSERVER

ZendServer installs PHP CLI as part of the installation, and, as is customary for PHP 4.3+, includes a PEAR installer. It’s a good idea to install PEAR before installing PHPUnit as per the recommendation here.

  1. Open Zend\ZendServer\bin\go-pear.bat and change the line:
    %PHP_BIN% -d output_buffering=0 -d PEAR\go-pear.pharto (see PHAR Runtime Configuration):%PHP_BIN% -d output_buffering=0 -d phar.require_hash=0 PEAR\go-pear.pharThis will stop the fatal error:phar...does not have a signatureas by default PHP will not process a PHAR archive without a signature.
  2. Run go-pear.bat

    For Windows 7, make sure you open the console as an administrator, or the installer won’t be able to create some folders in the default install locations. At the command prompt enter:#cd \program files\zend\zendserver\bin
    # go-pear.bat
    and you should get something like this (I selected the system-wide install and accepted the default locations):install pear

    If the installation succeeded, you should see something like:end of pear install

    Note that c:\program files\zend\zendserver\bin is already in Windows Path, thanks to ZendServer.

  3. Update PEAR Trying to install PHPUnit at this point will fail, complaining of an outdated installer. Update the PEAR package:# pear channel-update pear.php.net
    # pear upgrade pear
  4. Add the PHPUnit channels – see Installing PHPUnit
    # pear channel-discover pear.phpunit.de
    # pear channel-discover pear.symfony-project.com
  5. Finally install PHPUnit# pear install phpunit/PHPUnitIf the installation runs successfully you should see:installing phpunit

You can check your PHPUnit by running it from the command prompt:# phpunit

——————

As of this writing I’m using Zend Server CE version 5.1.0 for my WAMP stack on a Windows 7 64-bit machine.  I was recently researching some open source tools that required PEAR yet for some reason I had not installed that yet.  The PEAR installation was fairly straight forward but there were a couple small detours.  I’ve documented my PEAR installation here for reference.

  1. Go to http://pear.php.net/go-pear, copy all of the contents on this page and save the file as go-pear.php in your C:\Program Files (x86)\Zend\Apache2\htdocs\ directory.
  2. Open your web browser and go to http://localhost/go-pear.php.  Click the “Next >>” link.  (NOTE: As of this writing the Go-PEAR installer is at version 1.1.6 and I’m installing PEAR version 1.9.4.)
  3. Set the Installation prefix ($prefix) field to C:\Program Files (x86)\Zend\ZendServer\bin\  (NOTE: Don’t forget the trailing slash “\” after bin.)
  4. Scroll down and click the Install button.
  5. After the PEAR installation is complete you need to add the PEAR installation path to your Windows “PATH” environment variables.  Right click on My Computer and select Properties.  Go to Advanced system settings -> Environment Variables -> under System variables scroll down to the Variable “Path”, highlight it, and click Edit.  At the end of this string add “;C:\Program Files (x86)\Zend\ZendServer\bin\PEAR”  (NOTE: Do not include the double quotes and be sure not to add a trailing slash “\” after PEAR.)  Click OK -> OK -> OK.
  6. Open your php.ini file with Notepad or any other text editor.  Your php.ini file should be located here C:\Program Files (x86)\Zend\ZendServer\etc\php.ini.  You need to add the PEAR path to the include path.[Zend]include_path=”.;C:\Program Files (x86)\Zend\ZendServer\share\ZendFramework\library;C:\Program Files (x86)\Zend\ZendServer\bin\PEAR”
  7. Save your php.ini file and restart your computer.
  8. After your computer is restarted open up the command prompt.  On the command line enter “pear list” (NOTE: no double quotes)  You should see the stable version listed for your PEAR install.  If you don’t then you may be doing something different on your machine versus mine with regards to the installation.  Check all paths in this reference to make sure they are correct.
  9. If you’re getting a PHP Error “Invalid configuration directive” when running your pear command then you need to make a small change to your pear.bat file.  This file can be found in your C:\Program Files (x86)\Zend\ZendServer\bin\ directory.  Scroll down to the :RUN section and modify the line with the PHP directive “-d include_path”.  Add a “.;” before “%PHP_PEAR_INSTALL_DIR%” (NOTE: no double quotes)  So, your line should look like this “include_path=’.;%PHP_PEAR_INSTALL_DIR%’”.  Save your pear.bat file and rerun your pear list command.

… and that’s it.  You now have PEAR installed and can use all of it’s functionality.

If you thought this post was helpful and would like to give back to the community then click on the “Donate” button and make a donation. Thank you.