Xaop PHP高性能的AOP扩展

功能特色

基于对象的文档注解AOP模式

方法注入AOP模式

属性AOP模式

系统指令及其含义

xaop.method_prefix

AOP文档注解需要排除的方法前缀,也就是说如果方法以此前缀开头,那么就会跳过AOP解析

xaop.aop_mode

AOP工作模式,可选值: 1(正常模式) 2(文档注解AOP) 3(方法注入AOP)

xaop.property_aop

属性AOP模式开启状态,1(正常模式) 2(属性AOP模式开启)

安装

git clone https://github.com/liqiongfan/xaop.git

cd xaop

sudo ./install

指令配置含义:

; 1 不启用AOP

; 2 文档注解AOP模式

; 3 方法注入AOP模式

xaop.aop_mode = 2

; 是否配置过滤方法前缀

; 此模式仅在 文档注解AOP模式生效

xaop.method_prefix = ""

; 是否开启属性AOP模式

; 1 不开启 2 开启

xaop.property_aop = 2

1、方法注入AOP模式:

class Xaop

{

public function beforeGetAction()

{

echo 'Before GET Action' . PHP_EOL;

}

public function afterGetAction()

{

echo 'After GET Action' . PHP_EOL;

}

public function getXaop()

{

echo 'getXaop method' . PHP_EOL;

}

}

注意:添加AOP方法的第二个参数支持 一个 模糊关键字*来替代零个或者多个字符,使用多个 * 会造成匹配失败,AOP失效。

下面示例一个使用两种前置AOP的姿势(其他类型的AOP类似):

闭包模式

Xaop::addBeforeAop(Xaop::class, "getXaop", function(){

echo 'Before';

});

类的方法模式,注意这里存在类的生成开销

Xaop::addBeforeAop(Xaop::class, "getXaop", [Xaop::class, "beforeGetAction"]);

[注意]:Xaop的环绕模式AOP仅在此模式下生效,文档注解模式不支持环绕模式AOP,环绕AOP的回调函数存在一个资源参数,表示原方法的OPCODE上下文,如果用户不执行 Xaop::exec($resOpCode)的话,那么就会自动丢失

闭包环绕AOP

Xaop::addAroundAop(Xaop::class, "getXaop", function($resCode){

echo 'Before';

Xaop::exec($resCode); // 如果不执行本行代码,原始方法不执行,并且会丢失

echo 'After';

});

类方法环绕AOP

class Xaop

{

public function actionBefore($resCode)

{

echo 'Before Code'. PHP_EOL;

Xaop::exec($resCode);

echo 'After Code' . PHP_EOL;

}

public function getXaop()

{

echo 'getXaop' . PHP_EOL;

}

}

Xaop::addAroundAop(Xaop::class, "getXaop", [Xaop::class, "actionBefore"]);

【注意】:如果同时指定了环绕AOP和其他的AOP,那么除了环绕AOP之外,其他的AOP都失效,以环绕AOP为准。

Xaop 支持 五种 AOP 模式,分别是 前置AOP(addBeforeAop)、后置AOP(addAfterAop)、后置返回AOP(addAfterReturnAop)、后置异常AOP(addAfterThrowAop)、环绕AOP(addAroundAop)

其中 环绕AOP 跟其他的 AOP 互斥,如果存在环绕 AOP ,系统将会优先以 环绕AOP 模式处理,并且 环绕AOP 回调函数存在一个参数: $xaopExec 的一个资源表示切入方法的上下文,环绕AOP模式下,如果不在环绕AOP方法内,调用 :Xaop::exec($xaopExec); 那么实际的方法将会丢失,不会调用,在环绕模式下,实际方法需要开发者自行调用,并且在同个回调方法内,调用多次 Xaop::exec($xaopExec);,仅生效一次,重复调用无效。如:

Xaop::addAroundAop(NULL, "__get*", function($exec){

echo '_before
';

Xaop::exec($exec); // 此处调用多次,Xaop自动拦截,只执行一次

echo '_after
';

});

2、对象属性AOP模式:

Xaop支持对象的属性AOP模式 :支持前置与后置切入,不实现环绕切入是因为开发者可以通过魔术方法进行处理,简单的示例:

class Swing

{

public $di = 'Xaop';

}

Xaop::addPropertyBeforeReadAop(Swing::class, 'di', function($object, $propertyName){

echo '_before';

});

Xaop::addPropertyAfterReadAop(Swing::class, 'di', function($object, $propertyName){

echo '_after';

});

// 调用属性

$swing = new Swing();

echo $swing->di;

// 输出如下:

_before Xaop _after

3、基于对象调用的文档注解AOP模式:

/**

* Class Swing

* @Aspect

*/

class Swing

{

function _magicGetBefore() {

echo '_magicGetBefore()' . PHP_EOL;

}

function _magicGetAfter() {

echo '_magicGetAfter()' . PHP_EOL;

}

function _magicSuccess() {

echo '_magicSuccess()' . PHP_EOL;

}

function _magicFailure() {

echo '_magicFailure()' . PHP_EOL;

}

/**

* @before( value="Swing._magicGetBefore" )

* @after( value="Swing._magicGetAfter" )

* @success( value="Swing._magicSuccess" )

*/

public function __get($name)

{

echo '__get' . PHP_EOL;

return true;

}

/**

* @before( value="Swing._magicGetBefore" )

* @after( value="Swing._magicGetAfter" )

* @failure( value="Swing._magicFailure" )

*/

public function __set($name, $value)

{

echo '__set' . PHP_EOL;

return false;

}

}

示例1:

// 调用 __get

$swing = new Swing();

$swing->di

输出结果如下:

_magicGetBefore()

__get

_magicSuccess()

_magicGetAfter()

示例2:

// 调用 __set

$swing = new Swing();

$swing->di = "di";

输出结果如下:

_magicGetBefore()

__set

_magicFailure()

_magicGetAfter()

Xaop 目前 基于对象的文档注解 AOP模式 ,如果使用 静态调用(self::|parent::|static::|class) 等都不会被捕捉,核心不进行捕捉的原因在于文档注解存在调用注解类的 input方法,而 input方法的第一个参数为类的对象,因此会额外增加一次对象的生成开销,为了减少对象生成开销,核心去除了静态方法的捕捉功能。

$swig = new Swing();

$swig->goodLists();

// 输出如下:

_before goodLists

文档注解支持 自定义注解 与扩展 内置注解:

自定义注解

自定义注解必须继承自 Xaop\Annotations 接口,并且实现 input 方法即可,如下示例自定义了一个 @Tag 注解:

namespace app;

use Xaop\Annotations;

class Tag implements Annotations {

function input($object, $annotations) {

var_dump($object);

foreach($annotations as $key => $val) {

echo $key . '=' . $val . PHP_EOL;

}

}

}

使用的时候只需要传入全名即可:

/**

* @Aspect

*/

class Swing

{

/**

* @app\Tag( money = 5000, user = "Xaop" )

*/

public function getMoney() {

}

}

内置注解

内置强大的七个专用注解: @api、@disable、@before 、@after、@success、@failure、 @deprecated

@api

开发 API 推荐使用,使用本注解,直接可以向客户端返回 JSON 或者 XML 数据,只需要在修饰的方法体返回数组数据即可,注解包含两个参数:

type 与 charset, 如下使用:

/**

*@api(type=JSON, charset=UTF-8)

*/

public function newLists() {

return [ ['12' => [xxx,xxx], ['23'=>[xxx,xxx] ];

}

或者

/**

*@api(type=xml, charset=UTF-8)

*/

public function newLists() {

return [ ['12' => [xxx,xxx], ['23'=>[xxx,xxx] ];

}

注意:参数名区分大小写,参数值不区分大小写。

@disable

使用本注解可以禁用类的方法,使用本注解修饰的方法,就不会调用,并且不会提示任何错误信息,直接返回,本注解不包含任何参数。

@before

前置通知,在方法之前执行本注解包含的方法:如:

@before(value="app\models\User.startTransaction")

使用场景:在执行业务代码逻辑之前开启事务支持

@after

后置通知,在方法之后执行本注解包含的方法:如:

@after(value="app\log\InvokeLog.record")

使用场景:在接口调用之后进行日志记录

@success

方法体返回 true 的时候调用的通知:如:

@success(value="app\models\User.commit")

使用场景:在业务逻辑代码执行成功之后提交事务

@failure

方法体返回 false之后调用的通知,如下:

@failure(value="app\models\User.rollback")

使用场景:在业务逻辑代码方法体返回失败的时候回滚事务

@deprecated

标注类的方法是过期方法,当调用此方法的时候,会提示一条 E_DEPRECATED 的警告信息,需要在 php.ini 文件中开启

@deprecated

aop框架 php,xaop: 支持三种模式的AOP框架,弥补PHPer的不足,并且自带了文档的解析类库,可以一并使用,性能极好,欢迎 STAR 与 FORK。...相关推荐

  1. java集合框架支持三种类型,Java集合框架(一)

    集合类存放于java.util包中,集合类存放的都是对象的引用,而非对象本身,出于表达上的便利,我们称集合中的对象就是指集合中对象的引用(reference) 集合类型主要有3种:set(集).lis ...

  2. 深入学习SAP UI5框架代码系列之七:控件数据绑定的三种模式 - One Way, Two Way和OneTime实现原理比较

    这是Jerry 2021年的第 8 篇文章,也是汪子熙公众号总共第 279 篇原创文章. 系列目录 (0) SAP UI5应用开发人员了解UI5框架代码的意义 (1) SAP UI5 module懒加 ...

  3. 分布式事务讲解 - TX-LCN分布式事务框架(含LCN、TCC、TXC三种模式)

    分布式事务讲解 - TX-LCN分布式事务框架(含LCN.TCC.TXC三种模式) 分布式事务系列博客: TX-LCN框架原理 LCN 原理及主要特点 代码实现 实现场景 创建数据库及表(三个数据库, ...

  4. Nacos支持三种部署模式

    Nacos支持三种部署模式 Nacos支持三种部署模式 单机模式 - 用于测试和单机试用. 集群模式 - 用于生产环境,确保高可用. 多集群模式 - 用于多数据中心场景. 单机模式下运行Nacos L ...

  5. MODBUS主站调试工具和MODBUS从站调试工具 支持RTU、TCP、UDP三种模式

    MODBUS调试工具 C#源码 包含MODBUS主站调试工具和MODBUS从站调试工具 支持RTU.TCP.UDP三种模式 开发环境VS 2012/2015/2017,.NET Framework 4 ...

  6. mysql traditional_MySQL 5.6 以上版本支持三种sql_mode模式:ANSI、TRADITIONAL和STRICT_TRANS_TABLES。...

    MySQL 5.0 以上版本支持三种sql_mode模式:ANSI.TRADITIONAL和STRICT_TRANS_TABLES. ANSI ANSI模式:宽松模式,对插入数据进行校验,如果不符合定 ...

  7. 聊聊呗|红包系统支持以下三种模式

    本文标题:聊聊呗|红包系统支持以下三种模式 常见的消费者扫码送红包,导购扫码送红包.代理商扫码送红包,都是通过扫码送红包营销活动,来实现企业的各种战略目标. 目标一:扫码送红包,激励导购卖货 导购是门 ...

  8. httpd的三种模式比较

    查看你的httpd使用了哪种模式: /usr/local/apache2/bin/httpd -V |grep 'Server MPM' 使用哪种模式,需要在编译的时候指定 --with-mpm=pr ...

  9. DDD分层架构的三种模式

    本文来说下DDD分层架构的三种模式 文章目录 概述 DDD 分层架构 模式一:四层架构 模式二:五层架构 模式三:六边形架构 本文小结 概述 在讨论DDD分层架构的模式之前,我们先一起回顾一下DDD和 ...

最新文章

  1. Task02:青少年软件编程(Scratch)等级考试模拟卷(二级)
  2. 如何使用Arthas定位线上 Dubbo 线程池满异常
  3. 二叉树的序列化和反序列化
  4. JS——样式类的添加
  5. ORA-01172,ORA-01151
  6. Matlab制作朱利表
  7. 最良心不过AMD AM4接口下一代可继续用
  8. 2349 Arctic Network prim最小生成树 基础
  9. MySQL INSERT ... ON DUPLICATE KEY UPDATE语句
  10. JS 新浪API获取IP归属地
  11. 《战舰世界》携手汉堡王开启“战舰堡胃战”主题活动
  12. 移远 BC95-B5
  13. 2021-2027全球与中国氢动力引擎市场现状及未来发展趋势
  14. STM32F103ZET6+IIC+SHT20温湿度传感
  15. k210应用8-使用DMAC实现存储器与存储器之间的高速数据传输
  16. 单反相机的传奇—佳能单反50年辉煌之路(前言)
  17. iVMS-4200 Vs区别_异地恋 VS 网恋,哪个更难坚持到最后?| Newth互动158
  18. CuteFTP 问题及 ftp 模式详解
  19. 跨平台iOS自动化测试工具——tidevice
  20. 扫地机器人扫水泥地板有用吗_39元的扫地机器人你见过么,还送块水泥板砖……...

热门文章

  1. web中的classpath 和 classpath*
  2. http://circles.arenaofthemes.com/
  3. RHEL5-4的启动流程介绍
  4. 用ISA Server做***路由代替专线
  5. 如何获取独立项目开发经验
  6. ZZULIOJ 1067:有问题的里程表
  7. 信息学奥赛一本通 2051:【例3.1】偶数
  8. 信息学奥赛一本通 1069:乘方计算 | OpenJudge NOI 1.5 13
  9. 幸运数字(洛谷-P3292)
  10. 15 FI配置-财务会计-将记账期间变式分配给公司代码