php中使用魔术方法实现aop
2017-02-24 14:42
267 查看
前段时间工作忙,没空写博客。其实有很多值得写的,如果之后有时间就会慢慢提上来。
最近在用php做后台。用的是某开源框架,我的主要工作是做http接口供前台调用。因为有了框架,写接口变得十分傻瓜。具体操作是,新建一个类,类名以Service结尾。之后编写的函数只要不是以下划线开头则可以直接被前台访问,例如:
我们项目组都这样写代码,于是我也这样写。但写久了之后,就觉得每个接口都需要进行权限校验和日志就显得过分冗余。对于多年的java开发者,aop的理念还是比较深入头脑。我便开始着手将鉴权和日志提出来公用。
因为是用的第三方框架,其他项目也在用这个框架,不便去修改,也怕修改引入其他问题。只能从这个类想办法。之前看了php的反射和魔术方法,觉得应该可以使用它们来解决这个问题。
解决思路很简单。使用__call方法拦截函数调用,在其中加入鉴权和日志,之后再重新调用目标方法。本身__call方法只能在调用方调用不存在或者没权限的方法时才会触发,我们需要把getOrder设为private或者protected。如下:
移到父类需要解决一些问题,1- 如何得到子类名, 2- 如何调用子类方法。
解决第一个问题可以使用get_called_class()函数,此函数是Core.php中的函数,获取调用当前方法的类名。我们可以在父类定义__call函数,子类不定义。调用者调用子类不存在的方法时会触发__call函数,由于子类没定义,自然就会到父类的__call方法,而get_called_class()函数就能得到子类的名称。
解决第二个问题,可以使用反射,获得子类的方法,将目标方法的访问权设置为公开,再进行调用。如下
父类:
如此一来,就相当优雅。子类只需要继承,没有其他多余的东西。唯一需要的就是把接口定义成private或者protected。
最近在用php做后台。用的是某开源框架,我的主要工作是做http接口供前台调用。因为有了框架,写接口变得十分傻瓜。具体操作是,新建一个类,类名以Service结尾。之后编写的函数只要不是以下划线开头则可以直接被前台访问,例如:
<?php class OrderService { function getOrder() { //权限校验 //日志 } } ?>
我们项目组都这样写代码,于是我也这样写。但写久了之后,就觉得每个接口都需要进行权限校验和日志就显得过分冗余。对于多年的java开发者,aop的理念还是比较深入头脑。我便开始着手将鉴权和日志提出来公用。
因为是用的第三方框架,其他项目也在用这个框架,不便去修改,也怕修改引入其他问题。只能从这个类想办法。之前看了php的反射和魔术方法,觉得应该可以使用它们来解决这个问题。
解决思路很简单。使用__call方法拦截函数调用,在其中加入鉴权和日志,之后再重新调用目标方法。本身__call方法只能在调用方调用不存在或者没权限的方法时才会触发,我们需要把getOrder设为private或者protected。如下:
<?php class OrderService { function __call($name, $arguments){ if (!method_exists($this, $name)){ return; } //权限校验 //日志 $this->$name(); } private function getOrder() { } } ?>这下好了,所有之后新加在OrderService里面的接口都会经过权限校验和日志了。但是,这样做并不优雅,之后新建的其他类都需要做一下这个工作。于是考虑将此逻辑移到父类。
移到父类需要解决一些问题,1- 如何得到子类名, 2- 如何调用子类方法。
解决第一个问题可以使用get_called_class()函数,此函数是Core.php中的函数,获取调用当前方法的类名。我们可以在父类定义__call函数,子类不定义。调用者调用子类不存在的方法时会触发__call函数,由于子类没定义,自然就会到父类的__call方法,而get_called_class()函数就能得到子类的名称。
解决第二个问题,可以使用反射,获得子类的方法,将目标方法的访问权设置为公开,再进行调用。如下
父类:
<?php class AopParentService { function __call($name, $arguments) { //判断方法是否存在 if (!method_exists(get_called_class(), $name)){ return; } //判断权限 //日志 //调用方法 $class = new ReflectionClass(get_called_class()); $instance = $class->newInstance(); $method = $class->getMethod($name); $method->setAccessible(true); return $method->invoke($instance); } } ?>子类:
<?php class OrderService extends AopParentService { private getOrder(){ //处理数据 return order; } } ?>
如此一来,就相当优雅。子类只需要继承,没有其他多余的东西。唯一需要的就是把接口定义成private或者protected。
相关文章推荐
- PHP利用魔术方法实现准AOP
- PHP利用魔术方法实现准AOP
- 使用PHP魔术方法实现属性的set、get方法
- php 魔术方法使用说明
- dojo的connect方法使用与源代码剖析(在JavaScript里实现类似AOP的效果)
- 使用php实现下载生成某链接快捷方式的解决方法
- php通过魔术方法__call实现类函数重载
- PHP中使用cURL实现Get和Post请求的方法
- php 魔术方法使用说明
- 详解php的魔术方法__get()和__set()使用介绍
- php教程之魔术方法的使用示例(php魔术函数)
- 使用ASM4.0实现AOP的功能,监控每个方法的执行时间
- PHP中用魔术方法__autoload()实现类的自动加载
- PHP中使用cURL实现Get和Post请求的方法
- dojo的connect方法使用与源代码剖析(在JavaScript里实现类似AOP的效果)
- php 魔术方法使用说明
- dojo的connect方法使用与源代码剖析(在JavaScript里实现类似AOP的效果)
- 实现Nginx中使用PHP-FPM时记录PHP错误日志的配置方法
- PHP中使用cURL实现Get和Post请求的方法