PHP的AOP扩展雨后春笋般的复活啦

Xaop PHP高性能的AOP扩展

功能特色

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

  • 方法注入AOP模式

框架
  • Yaf

  • CSpeed

  • Xannotation

  • Phalcon

系统指令及其含义
  • xaop.method_prefix

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

  • xaop.aop_mode

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

安装

git clone https://github.com/liqiongfan/xaop.git
  ​
  cd xaop
  ​
  /usr/local/path_to_php/bin/phpize
  ​
  ./configure --with-php-config=/usr/local/path_to_php/bin/php-config
      
  make -j && sudo make install
启用对应功能扩展需要在 php.ini 文件配置指令: xaop.aop_mode,如下:
; To enable the AOP mode
  ; 1 不启用AOP
  ; 2 文档注解AOP模式
  ; 3 方法注入AOP模式
  xaop.aop_mode = 2
1、方法注入AOP模式:
<?php
      
  class Swing
  {
      public function __get($name)
      {
          echo '__get<br>';
      }
  ​
  ​
      public function __set($name, $value)
      {
          echo '__set<br>';
      }
  ​
  }
  ​
  // 注入前置AOP
  Xaop::addBeforeAop(Swing::class, "__get*", function(){
      echo '__get*_before<br>';
  });
  ​
  // 注入后置AOP
  Xaop::addAfterAop(Swing::class, "__get*", function(){
      echo '__get*_after<br>';
  });
  ​
  // 注入后置返回AOP(当方法返回不是null的内容后,此AOP生效)
  Xaop::addAfterReturnAop(Swing::class, "__get*", function(){
      echo '__get*_after_return<br>';
  });
  ​
  // 注入后置抛出异常AOP(当方法抛出异常的时候,此AOP生效)
  Xaop::addAfterThrowAop(Swing::class, "__get*", function(){
      echo '__get*_after_throw<br>';
  });
  ​
  // 注入环绕AOP(注意环绕AOP与其他的AOP不可同用,存在环绕AOP的情况下,一切以环绕AOP为准)
  //Xaop::addAroundAop(NULL, "__get*", function($exec){
  //    echo '_before<br>';
  //    var_dump(Xaop::exec($exec));
  //    echo '_after<br>';
  //});
<?php
      
  $swing = new Swing();
  $swing->di;
  ​
  //输出结果如下
  __get*_before
  __get
  __get*_after

注意

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<br>';
      Xaop::exec($exec); // 此处调用多次,Xaop自动拦截,只执行一次
      echo '_after<br>';
  });
2、基于对象调用的文档注解AOP模式:
<?php
  ​
  /**
   * 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方法的第一个参数为类的对象,因此会额外增加一次对象的生成开销,为了减少对象生成开销,核心去除了静态方法的捕捉功能。

因为基于 Zend 执行引擎,所以不需要使用代理对象完成切面,直接调用方法 即可:

<?php
  ​
  $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;
          }
      }
  }

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

<?php
  ​
  /**
   * @Aspect
   */
  class Swing
  {
      /**
       * @app\Tag( money = 5000, user = "Xaop" )
       */
      public function getMoney() {
          
      }
  }
  • 内置注解

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

    1. @api

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

      typecharset, 如下使用:

      /**
         *@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] ];
        }

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

    2. @disable

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

    3. @before

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

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

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

    4. @after

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

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

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

    5. @success

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

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

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

    6. @failure

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

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

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

    7. @deprecated

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

      @deprecated

转载于:https://my.oschina.net/cspeed/blog/2054797

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
runkit扩展库是pecl的php扩展的一部分。 runkit提供了一套函数,能让你动态的修改任意类和函数的定义。根据php提供的反射机制,得到所有用户定义的类和函数,你可以把他们重命名然后再用原来的函数名/方法名重新添加一个方法/函数,重新添加的函数的函数体,你可以任意定义---用字串儿,当你的php脚本开始执行时,你要最先 调用你设计的这个“钩子”,把所有的类的方法或函数都"钩"一遍,这样,不管是哪个方法在执行的时候,都会执行到你提供的切入点上的代码. 不过,在使用runkit的时候,还遇到一个问题:就是大小写的问题,起初,在“钩”方法的时候,有的方法带有大写字母,通过反射得到的方法名 也是区分大小写的,但是,应该在php解析执行函数的时候,还是把所有方法名或函数名都转换成了小写,这样,在用runkit提供的动态添加方法的函数添 加方法时,也忠实方法原来的命名,带着大写字母就直接添进去了,结果,再调用被钩的方法的时候,报错,死活也找不着原来的那个方法了,原因应该就 是,php在解析执行类、方法、函数时,应该都是按小写形式来定位的方法和函数的,而用runkit直接添加进去的带有大写字母的函数,自然就会找不到, 所以,在用runkit添加方法的时候,要把方法名都转换成小写,这样php就可以找到这个方法了,函数应该也是同样的. 示例代码: php.ini:runkit.superglobal=foo,bartest.php:function testme() {  echo "Foo is $foo\n";  echo "Bar is $bar\n";  echo "Baz is $baz\n";}$foo = 1;$bar = 2;$baz = 3;testme();输出:Foo is 1Bar is 2Baz is兼容性: PHP 4.2 or greater 标签:Runkit

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值