# CBD模式

[上一页](# "上一页")[下一页](# "下一页")

ThinkPHP从3.0版本开始引入了全新的**CBD(核心Core+行为Behavior+驱动Driver)**架构模式,因为从底层开始,框架就采用核心+行为+驱动的架构体系,核心保留了最关键的部分,并在重要位置设置了标签用以标记,其他功能都采用行为扩展和驱动的方式组合,开发人员可以根据自己的需要,对某个标签位置进行行为扩展或者替换,就可以方便的定制框架底层,也可以在应用层添加自己的标签位置和添加应用行为。而标签位置类似于AOP概念中的“切面”,行为都是围绕这个“切面”来进行编程。

### Core(核心)

ThinkPHP的核心部分包括核心函数库、惯例配置、核心类库(包括基础类和内置驱动及核心行为),这些是ThinkPHP必不可少的部分。

~~~

ThinkPHP/Common/functions.php // 核心函数库

ThinkPHP/Conf/convention.php // 惯例配置文件

ThinkPHP/Conf/debug.php // 惯例调试配置文件

ThinkPHP/Mode/common.php // 普通模式定义文件

ThinkPHP/Library/Think // 核心类库包

ThinkPHP/Library/Behavior // 系统行为类库

ThinkPHP/Library/Think/App.class.php // 核心应用类

ThinkPHP/Library/Think/Behavior.class.php // 基础行为类

ThinkPHP/Library/Think/Cache.class.php // 核心缓存类

ThinkPHP/Library/Think/Controller.class.php // 基础控制器类

ThinkPHP/Library/Think/Db.class.php // 数据库操作类

ThinkPHP/Library/Think/Dispatcher.class.php // URL解析调度类

ThinkPHP/Library/Think/Exception.class.php // 系统基础异常类

ThinkPHP/Library/Think/Hook.class.php // 系统钩子类

ThinkPHP/Library/Think/Log.class.php // 系统日志记录类

ThinkPHP/Library/Think/Model.class.php // 系统基础模型类

ThinkPHP/Library/Think/Route.class.php // 系统路由类

ThinkPHP/Library/Think/Storage.class.php // 系统存储类

ThinkPHP/Library/Think/Template.class.php // 内置模板引擎类

ThinkPHP/Library/Think/Think.class.php // 系统引导类

ThinkPHP/Library/Think/View.class.php // 系统视图类

~~~

Behavior目录下面是系统内置的一些行为类库,内置驱动则分布在各个不同的驱动目录下面(参考下面的驱动部分)。

### Driver(驱动)

3.2在架构设计上更加强化了驱动的设计,替代了之前的引擎和模式扩展,并且改进了行为的设计,使得框架整体更加灵活,并且由于在需要写入数据的功能类库中都采用了驱动化的设计思想,所以使得新的框架能够轻松满足分布式部署的需求,对云平台的支持可以更简单的实现了。因此,在新版的扩展里面,已经取消了引擎扩展和模式扩展,改成配置不同的应用模式即可。

驱动包括

~~~

ThinkPHP/Library/Think/Cache/Driver // 缓存驱动类库

ThinkPHP/Library/Think/Db/Driver // 数据库驱动类库

ThinkPHP/Library/Think/Log/Driver // 日志记录驱动类库

ThinkPHP/Library/Think/Session/Driver // Session驱动类库

ThinkPHP/Library/Think/Storage/Driver // 存储驱动类库

ThinkPHP/Library/Think/Template/Driver // 第三方模板引擎驱动类库

ThinkPHP/Library/Think/Template/TagLib // 内置模板引擎标签库扩展类库

~~~

### Behavior(行为)

行为(Behavior)是ThinkPHP扩展机制中比较关键的一项扩展,行为既可以独立调用,也可以绑定到某个标签(位)中进行侦听。这里的行为指的是一个比较抽象的概念,你可以想象成在应用执行过程中的一个动作或者处理,在框架的执行流程中,各个位置都可以有行为产生,例如路由检测是一个行为,静态缓存是一个行为,用户权限检测也是行为,大到业务逻辑,小到浏览器检测、多语言检测等等都可以当做是一个行为,甚至说你希望给你的网站用户的第一次访问弹出Hello,world!这些都可以看成是一种行为,行为的存在让你无需改动框架和应用,而在外围通过扩展或者配置来改变或者增加一些功能。

而不同的行为之间也具有位置共同性,比如,有些行为的作用位置都是在应用执行前,有些行为都是在模板输出之后,我们把这些行为发生作用的位置称之为**标签(位)**,也可以称之为钩子,当应用程序运行到这个标签的时候,就会被拦截下来,统一执行相关的行为,类似于AOP编程中的“切面”的概念,给某一个标签绑定相关行为就成了一种类AOP编程的思想。

##### 系统标签位

系统核心提供的标签位置包括(按照执行顺序排列):

- app_init 应用初始化标签位

- module_check 模块检测标签位(**3.2.1版本新增**)

- path_info PATH_INFO检测标签位

- app_begin 应用开始标签位

- action_name 操作方法名标签位

- action_begin 控制器开始标签位

- view_begin 视图输出开始标签位

- view_template 视图模板解析标签位

- view_parse 视图解析标签位

- template_filter 模板解析过滤标签位

- view_filter 视图输出过滤标签位

- view_end 视图输出结束标签位

- action_end 控制器结束标签位

- app_end 应用结束标签位

在每个标签位置,可以配置多个行为,行为的执行顺序按照定义的顺序依次执行。除非前面的行为里面中断执行了(某些行为可能需要中断执行,例如检测机器人或者非法执行行为),否则会继续下一个行为的执行。

除了这些系统内置标签之外,开发人员还可以在应用中添加自己的应用标签,在任何需要拦截的位置添加如下代码即可:

~~~

tag('my_tag'); // 添加my_tag 标签侦听

// 下面的写法作用一致

\Think\Hook::listen('my_tag');

~~~

tag函数用于设置某个标签位,可以传入并且只接受一个参数,如果需要传入多个参数,请使用数组,

~~~

tag('my_tag',$params); // 添加my_tag 标签侦听

~~~

该参数为引用传值,所以只能传入变量,因此下面的传值是错误的:

~~~

tag('my_tag','param'); // 添加my_tag 标签侦听

~~~

##### 核心行为

系统的很多核心功能也是采用行为扩展组装的,对于满足项目日益纷繁复杂的需求和定制底层框架提供了更多的方便和可能性。

核心行为位于 `ThinkPHP/Behavior/` 目录下面,框架核心内置的行为包括如下:

| 行为名称 | 说明 | 对应标签位置 |

|-----|-----|-----|

| BuildLite | 生成Lite文件(3.2.1版本新增) | app_init |

| ParseTemplate | 模板文件解析,并支持第三方模板引擎驱动 | view_parse |

| ShowPageTrace | 页面Trace功能行为,完成页面Trace功能 | view_end |

| ShowRuntime | 运行时间显示行为,完成运行时间显示 | view_filter |

| TokenBuild | 令牌生成行为,完成表单令牌的自动生成 | view_filter |

| ReadHtmlCache | 读取静态缓存行为 | app_init |

| WriteHtmlCache | 生成静态缓存行为 | view_filter |

##### 行为定义

自定义的扩展行为可以放在核心或者应用目录,只要遵循命名空间的定义规则即可。行为类的命名采用:行为名称(驼峰法,首字母大写)+Behavior 行为类的定义方式如下:

~~~

namespace Home\Behavior;

use Think\Behavior;

class TestBehavior extends Behavior {

// 行为扩展的执行入口必须是run

public function run(&$params){

if(C('TEST_PARAM')) {

echo 'RUNTEST BEHAVIOR '.$params;

}

}

}

~~~

**3.2.1版本**开始,行为类的定义无需继承`Think\Behavior`类,所以,上面的定义可以简化为:

~~~

namespace Home\Behavior;

class TestBehavior {

// 行为扩展的执行入口必须是run

public function run(&$params){

if(C('TEST_PARAM')) {

echo 'RUNTEST BEHAVIOR '.$params;

}

}

}

~~~

行为类必须定义执行入口方法`run`,由于行为的调用机制影响,run方法不需要任何返回值,所有返回都通过引用返回。

> run方法的参数只允许一个,但可以传入数组。

##### 行为绑定

行为定义完成后,就需要绑定到某个标签位置才能生效,否则是不会执行的。

我们需要在应用的行为定义文件`tags.php`文件中进行行为和标签的位置定义,格式如下:

~~~

return array(

'标签名称1'=>array('行为名1','行为名2',...),

'标签名称2'=>array('行为名1','行为名2',...),

);

~~~

标签名称包括我们前面列出的系统标签和应用中自己定义的标签名称,比如你需要在app_init标签位置定义一个CheckLangBehavior行为类的话,可以使用:

~~~

return array(

'app_init'=>array('Home\Behavior\CheckLang'),

// **如果是3.2.1版本 需要改成(后面不再重复说明)**

// 'app_init'=>array('Home\Behavior\CheckLangBehavior'),

);

~~~

可以给一个标签位定义多个行为,行为的执行顺序就是定义的先后顺序,例如:

~~~

return array(

'app_init'=>array(

'Home\Behavior\CheckLang',

'Home\Behavior\CronRun'

),

);

~~~

默认情况下tags.php中定义的行为会并入系统行为一起执行,也就是说如果系统的行为定义中app_init标签中已经定义了其他行为,则会首先执行系统行为扩展中定义的行为,然后再执行项目行为中定义的行为。例如:系统行为定义文件中定义了:

~~~

'app_begin' => array(

'Behavior\ReadHtmlCache', // 读取静态缓存

),

~~~

而应用行为定义文件有定义:

~~~

'app_begin' => array(

'Home\Behavior\CheckModule',

'Home\Behavior\CheckLang',

),

~~~

则最终执行到app_begin标签(位)的时候,会依次执行:

~~~

Library\Behavior\ReadHtmlCache

Home\Behavior\CheckModule

Home\Behavior\CheckLang

~~~

三个行为(除非中间某个行为有中止执行的操作)。

如果希望应用的行为配置文件中的定义覆盖系统的行为定义,可以改为为如下方式:

~~~

'app_begin' => array(

'Home\Behavior\CheckModule',

'Home\Behavior\CheckLang',

'_overlay' => true,

),

~~~

则最终执行到app_begin标签(位)的时候,会依次执行下面两个行为:

~~~

Home\Behavior\CheckModule

Home\Behavior\CheckLang

~~~

应用行为的定义没有限制,你可以把一个行为绑定到多个标签位置执行,例如:

~~~

return array(

'app_begin'=>array('Home\Behavior\Test'), // 在app_begin 标签位添加Test行为

'app_end'=>array('Home\Behavior\Test'), // 在app_end 标签位添加Test行为

);

~~~

##### 单独执行

行为的调用不一定要放到标签才能调用,如果需要的话,我们可以在控制器中或者其他地方直接调用行为。例如,我们可以把用户权限检测封装成一个行为类,例如:

~~~

namespace Home\Behavior;

use Think\Behavior;

class AuthCheckBehavior extends Behavior {

// 行为扩展的执行入口必须是run

public function run(&$return){

if(C('USER_AUTH_ON')) {

// 进行权限认证逻辑 如果认证通过 $return = true;

// 否则用halt输出错误信息

}

}

}

~~~

定义了AuthCheck行为后,我们可以在控制器的_initialize方法中直接用下面的方式调用:

~~~

B('Home\Behavior\AuthCheck');

// 3.2.1版本中需要改成

B('Home\Behavior\AuthCheckBehavior');

~~~

[上一页](# "上一页")[下一页](# "下一页")

php cbd架构,CBD模式相关推荐

  1. 大数据架构和模式(一)——大数据分类和架构简介

    概述 大数据可通过许多方式来存储.获取.处理和分析.每个大数据来源都有不同的特征,包括数据的频率.量.速度.类型和真实性.处理并存储大数据时,会涉及到更多维度,比如治理.安全性和策略.选择一种架构并构 ...

  2. .NET架构与模式探索

    什么是架构 软件体系结构通常被称为架构,指可以预制和可重构的软件框架结构.架构尚处在发展期,对于其定义,学术界尚未形成一个统一的意见,而不同角度的视点也会造成软件体系结构的不同理解,以下是一些主流的标 ...

  3. 这 3 种 DDD 分层架构的模式,你掌握了么?

    -     前言    - 在讨论DDD分层架构的模式之前,我们先一起回顾一下DDD和分层架构的相关知识. -     DDD 的基本概念     - DDD(Domain DrivenDesign, ...

  4. 微服务架构通讯模式架构分析

    微服务架构通讯模式架构分析) 目录 概 述 小结 参考资料和推荐阅读 LD is tigger forever,CG are not brothers forever, throw the pot a ...

  5. 大数据架构和模式(五):利用大数据识别保险行业中的欺诈业务案例

    大数据架构和模式(五):利用大数据识别保险行业中的欺诈业务案例 [复制链接]     pig2 1232 主题 2069 帖子 1万 积分 超级版主 积分 11358 收听TA 发消息 电梯直达 楼主 ...

  6. 武汉理工大学 大数据架构与模式期末复习

    武汉理工大学 大数据架构与模式期末复习 在发现学长学姐们上一届是大作业结课而我们是考试结课之后整理复习的结果,可能不会很全,感觉最后老师稍微划知识点的时候没有为难我们(),总之大体是这么些考试内容,也 ...

  7. MVC三层架构的模式

    大家好,今天给大家分享一下MVC 三层架构的模式 首先你要知道,所谓的MVC就是一种面向于javaee企业级开发的设计模式 这里要强调一点,MVC 不是一种技术,不是一种像spring 那样的框架,它 ...

  8. 架构师是如何炼成的?以天猫APP架构开发模式升级工程为例

    在集团大数据.算法的背景下,猫客(天猫客户端)首页率先从2015年的坑位运营走向2016年的全面个性化,猫客首页个性化业务点多达50多处,个性化场景大部分通过通过Aladdin(天猫推荐)接入TPP( ...

  9. 大型Javascript应用架构的模式(译文)

    附上翻译好的word文件 http://files.cnblogs.com/lizhug/Patterns_For_Large-Scale_JavaScript_Application_Archite ...

最新文章

  1. sklearn基本回归方法
  2. VTK:可视化之MoveCamera
  3. 从外网给新建的Exchange 2007/2010分发通讯组发邮件失败
  4. python数字转换_Python实现中文数字转换为阿拉伯数字的方法示例
  5. Python全栈开发:web框架们
  6. 【其他】学习通下载任务点里的PPT三步走
  7. idea社区版使用maven运行web项目
  8. ArrayList以及List的常用方法
  9. matlab时域转换成频域_频域的弦,时域的箭(1)
  10. 短视频矩阵系统H5形式视频分享如何开发?
  11. 计算机网络与互联网(三)
  12. 360无线wifi路由器连接到服务器,把360无线路由器设置为二级路由器 | 192路由网...
  13. 主角叫张四的Java游戏_第一卷 :仙剑世界中的黑化张四 123 贝爷附体
  14. 大文件上传NeatUpload简单用法 (转)
  15. 2016弱校联盟十一专场10.2部分题解
  16. 四种电子取证软件的比较
  17. AWS Route 53里使用外部系统的域名
  18. Leetcode典型题解答和分析、归纳和汇总——T155(最小栈)
  19. 深圳市社会组织信息平台爬虫获取信息
  20. 多影响因素下,到底咋归因?!

热门文章

  1. .NET Core UI框架Avalonia
  2. 基于Emgu CV+百度人脸识别,实现视频动态 人脸抓取与识别
  3. Web前端知识体系精简
  4. 如何迁移#SNMP到.NET Core平台的一些体会
  5. oracle optimizer_features_enable,Oracle Optimizer:迁移到使用基于成本的优化器—–系列2.1-数据库专栏,ORACLE...
  6. 拆分备份(还原)比较大的数据库为多个bak文件
  7. Android Studio开发基础之Context用法说明
  8. C语言试题十五之编写函数void function(int x,int pp[],int *n),求出能整除x且不是偶数的各整数,并按从小到大的顺序放在pp所指的数组中,这些除数的个数通过形参n返回
  9. Android studio之如何快速查看页面的布局
  10. C和指针之反转字符串