php中使用魔术方法实现aop

来源:互联网 发布:基因大数据公司 编辑:程序博客网 时间:2024/06/03 12:04

前段时间工作忙,没空写博客。其实有很多值得写的,如果之后有时间就会慢慢提上来。

最近在用php做后台。用的是某开源框架,我的主要工作是做http接口供前台调用。因为有了框架,写接口变得十分傻瓜。具体操作是,新建一个类,类名以Service结尾。之后编写的函数只要不是以下划线开头则可以直接被前台访问,例如:

<?phpclass OrderService {    function getOrder() {        //权限校验        //日志            }}?>

我们项目组都这样写代码,于是我也这样写。但写久了之后,就觉得每个接口都需要进行权限校验和日志就显得过分冗余。对于多年的java开发者,aop的理念还是比较深入头脑。我便开始着手将鉴权和日志提出来公用。

因为是用的第三方框架,其他项目也在用这个框架,不便去修改,也怕修改引入其他问题。只能从这个类想办法。之前看了php的反射和魔术方法,觉得应该可以使用它们来解决这个问题。

解决思路很简单。使用__call方法拦截函数调用,在其中加入鉴权和日志,之后再重新调用目标方法。本身__call方法只能在调用方调用不存在或者没权限的方法时才会触发,我们需要把getOrder设为private或者protected。如下:

<?phpclass 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()函数就能得到子类的名称。

解决第二个问题,可以使用反射,获得子类的方法,将目标方法的访问权设置为公开,再进行调用。如下

父类:

<?phpclass 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);    }}?>
子类:

<?phpclass OrderService extends AopParentService {private getOrder(){//处理数据return order;}}?>

如此一来,就相当优雅。子类只需要继承,没有其他多余的东西。唯一需要的就是把接口定义成private或者protected。




0 0
原创粉丝点击