php magic method 的具体应用和 phpdoc 结合

关于 Magic Methods 的介绍自行查阅官方文档,这里不再赘述。
http://php.net/manual/en/lang...

使用 phpstorm 的同学注意了,如果在我们的代码中使用到了 php 中相关的魔术方法,需要在 php 文件中指明告诉 phpstorm 应该如何来跟踪变量属性。下面我们来具体实践分析。

假设现在我有一个 php 的基类名为 BaseController.php

<?php

namespace Conduit\Controllers;

use Psr\Container\ContainerInterface;

/**
 * Class BaseController
 * @package Conduit\Controllers
 */
class BaseController
{
    /** @var \Interop\Container\ContainerInterface */
    protected $container;

    /** @var \Conduit\Services\Auth\Auth */
    private $auth;

    /** @var \Conduit\Validation\Validator */
    private $validator;

    /** @var \League\Fractal\Manager */
    private $fractal;

    /**
     * BaseController constructor.
     * @param ContainerInterface $container
     */
    public function __construct (ContainerInterface $container)
    {
        $this->container = $container;
    }

    /**
     * @param $name
     * @return mixed
     */
    public function __isset ($name)
    {
        return $this->container->{$name};
    }

    /**
     * @param $name
     * @param $value
     * @return mixed
     */
    public function __set ($name, $value)
    {
        return $this->container->{$name} = $value;
    }

    /**
     * @param $name
     * @return mixed
     */
    public function __get ($name)
    {
        return $this->container->{$name};
    }
}

同时还有一个 CompanyController.php 的类文件继承的该 BaseController.php

<?php

namespace Conduit\Controllers\Company;

class CompanyController extends BaseController
{
  public function getSpecialSubs (Request $request, Response $response)
    {
        $couser = $this->auth->requestUser($request);
    }
}

这里我们主要要关注的点就是 $this->auth 这个,因为这个才是我们今天要将的重点。这里再说一点,列举的代码是基于 slim framework 中的一部分,有不了解 slim framework 的可以先了解下。当然不了解也不影响我们今天要说的这一点。下面是正题。

这里我想实现的效果就是当我在 CompanyController.php 的方法中要访问这个容器(container)中的 auth,可以直接使用 $this->auth,然后就可以操作auth中的方法和属性等。

那上面我列举的方式是使用 php 语言的 magic methods 这个特性,那有同学就要说了,我直接指定岂不是也可以,答案是肯定的。

<?php

namespace Conduit\Controllers;

use Interop\Container\ContainerInterface;

class BaseController
{

    /** @var \Interop\Container\ContainerInterface */
    protected $container;

    /** @var \Conduit\Services\Auth\Auth */
    protected $auth;

    /** @var \Conduit\Validation\Validator */
    protected $validator;

    /** @var \Illuminate\Database\Capsule\Manager */
    protected $db;

    /** @var \League\Fractal\Manager */
    protected $fractal;

    /** @var \SlimSession\Helper */
    protected $session;

    /**
     * BaseController constructor.
     *
     * @param \Interop\Container\ContainerInterface $container
     */
    public function __construct(ContainerInterface $container)
    {
        $this->container = $container;
        $this->auth      = $this->container->get('auth');
        $this->validator = $this->container->get('validator');
        $this->fractal   = $this->container->get('fractal');
        $this->session   = $this->container->get('session');
        $this->db        = $this->container->get('db');
    }

效果就是这个样子。
确实这个样子可以实现,但没有利用到 magic method 这一魔术方法的特性。

这里着重说明的是这个我们定义的这些个类内部属性(这些成员属性必须通过phpdoc 的 @var 来指明你将要调用的成员属性是哪一个不然 phpstorm 不会识别提示)和 __get() 方法,当我们要访问不可访问的属性时会调用该方法。
那所以说 $auth | $validator | $db | fractal 等属性我们要在外部自动调用 __get() 方法来访问的话就必须将其访问修饰符设置为 private (php 中的三种修饰符 public protected private),这样我们在外部调用才能自动执行到 __get() 方法,达到同样的目的来调用 $this->auth 等

相关推荐