sele 发表于 2018-11-4 08:21:06

PHP laravel框架Redis门面的误用

  使用laravel的Redis时候乱用Facades踩了一个坑。。。
  判断Redis是否有某个key值得时候是可以这样写,
Redis::exists(key值)  因为使用了Redis门面,所以可以直接使用Redis::exists而不用先建立实例,就是文档中介绍的“Facade 基类使用魔术方法 __callStatic() 从你的门面中调用解析对象”。
  简单说就是没有exists这个静态方法,使用的时候会调用__callStatic(),建立实例:
$instance = static::getFacadeRoot();  使用Redis时候没有使用默认库0,所以按照文档上的例子指定服务,
  代码如下:
Redis::connection('users');  
Redis::exists(key值)
  这么做不能判断key值是否存在,测试以后发现第二行Redis::exists连接的仍然是默认库0。
  乱用Facades踩到坑了!
  我用的是laravel5.2,redis使用Predis。
  虽然Redis的Facades是一个静态代理,在使用时候"Redis::"也是静态唯一的,BUT connection并不是一个静态方法。
  Redis对应的Facades底层类 Illuminate\Redis\Database,connection方法如下
    /**  
   * Get a specific Redis connection instance.
  
   *
  
   * @paramstring$name
  
   * @return \Predis\ClientInterface|null
  
   */
  
    public function connection($name = 'default')
  
    {
  
      return Arr::get($this->clients, $name ?: 'default');
  
    }
  返回redis的链接实例。
  实际上$this->clients在Redis::调用__callStatic()建立实例的时候就已经初始化,
    /**  
   * Create a new Redis connection instance.
  
   *
  
   * @paramarray$servers
  
   * @return void
  
   */
  
    public function __construct(array $servers = [])
  
    {
  
      $cluster = Arr::pull($servers, 'cluster');
  
      $options = (array) Arr::pull($servers, 'options');
  
      if ($cluster) {
  
            $this->clients = $this->createAggregateClient($servers, $options);
  
      } else {
  
            $this->clients = $this->createSingleClients($servers, $options);
  
      }
  
    }
  打印发现 $this->clients 是将 redis的所有配置都初始化,当connection传入配置名的时候选择这个配置链接实例。
  虽然Redis实例是静态的,但是这个redis链接并不是。
  Redis::exists()更是和connection()方法无关,redis门面调用__callStatic()后会执行 Illuminate\Redis\Database 的exists方法,但是Database没有这个方法,所以触发__call()方法。
    /**  
   * Dynamically make a Redis command.
  
   *
  
   * @paramstring$method
  
   * @paramarray   $parameters
  
   * @return mixed
  
   */
  
    public function __call($method, $parameters)
  
    {
  
      return $this->command($method, $parameters);
  
    }
  而command() 方法使用的是default配置,也就是说Redis::所有方法除了connection都是默认库。
    /**  
   * Run a command against the Redis database.
  
   *
  
   * @paramstring$method
  
   * @paramarray   $parameters
  
   * @return mixed
  
   */
  
    public function command($method, array $parameters = [])
  
    {
  
      return call_user_func_array([$this->clients['default'], $method], $parameters);
  
    }
  所以要使用非默认配置的同一个redis链接时候必须保存redis实例,
$redis = Redis::connection("user");  
$redis->exists(key值);
  $redis->exists 不会调用database的command方法,redis继续使用connection选择的初始化链接,不会选择default配置。
  这其实还涉及到Predis底层的实现,已经超出我的理解范围。
  看完代码感觉还是有点晕。。。
  反正Redis门面还是不要乱用。


页: [1]
查看完整版本: PHP laravel框架Redis门面的误用