eloquent为何无需实例化就能调用类的方法?或者内部是怎么实现实例化的?

使用eloquent操作数据库,只需定义一个类就可以和数据库中的表关联

<?php

class Article extends \Eloquent {

protected $fillable = [];

}

如果想查找某条记录


$article = Article::find(2);

echo $article->title;

但问题是如果直接用 :: 来访问某个 function,无论这个 function 是否为 static,构造函数 __construct() 都不会被调用,那么创建对象实例在eloquent中是如何实现的呢?

阅读 4.7k
4 个回答

难道是用的Laravel 4
extends Eloquent

因为Laravel 5是
extends Model

Model 中 构造函数一般情况下是没用的,除非用于create/fill操作
Model.php没有的静态函数、非静态函数的调取方式如下:

function __call(){
    $query = $this->newQuery();  //这里new 的Builder

    return call_user_func_array([$query, $method], $parameters); //之后都是调取的Builder的对象,比如find,where
}

function __callStatic(){
    $instance = new static;
    return call_user_func_array([$instance, $method], $parameters); //调取上面的 __call
}

这是魔术方法,看手册,然后自己多想一下是怎么调用的。
这段的确很绕,但是看懂了也就明白了Model的运行机制

下面是github laravel5.3版本的实现方法。

/**
 * Handle dynamic static method calls into the method.
 *
 * @param  string  $method
 * @param  array  $parameters
 * @return mixed
 */
public static function __callStatic($method, $parameters)
{
    return (new static)->$method(...$parameters);
}

源码地址

另外这里使用static(实例化运行时的类,例如你的Article)和self(实例化Model)是有区别的。

暂时没翻过这个源码,但是我可以肯定的是他的做法。

public static function __callStatic($method, $params) {
    if(is_null(self::$article)) {
        self::$article = new Article();
    }

    call_user_func_array([self::$article, $method], $params);
}

怎么做到的?
1、php的魔术方法,当调用一个静态方法不存在的时候调用__callStatic这个函数。
2、__callStatic这个函数里面先判断是否有初始化这个Article,如果没有,那么初始化,返回实例

以上。

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题