二、Db类库场景分析
先从一个简单的案例进行解析,先来看一下数据库的数据。
然后来到控制器写一个简单的查询案例,在创建控制器之前先使用命令进行创建一个测试控制器。
在这个控制器进行简单的查询数据。
查询结果如下
在这个案例中,可以看到使用的是Db::query这种查询方式,接下来对于这种查询方式进行简单的剖析。
接着执行流程就会来到Db这个类,在这个类中可以看到关于当对象访问不存在的静态方法时,__callStatic()方法会被自动调用。
这个方法在之前门面的讲解中进行了深度讲解。
从上图可以看到当执行访问对象不存在的静态方法时会执行到call_user_func_array调用回调函数,并把一个数组参数作为回调函数的参数
接着代码就会执行到static::connect()这行代码,由于本类Db没有继承任何的类,所以对于static这个的使用就是调用本类。
如果当Db类继承了其它类那么就会有一定的区别,这个区别就是关于static关键字,给大家做的一点点冷门知识得补充,当一个类继承一个类时,在父类实用static关键字时,默认调用的子类的方法。
切换数据库连接
因为没有任何继承,所以会来到本类的connect这个方法。
在这个类里边首先会返回结果为数据库配置信息。
然后会从配置信息中获取到query这个索引,最终返回\think\db\Query这个字符串,这里一定要注意返回的是字符串,不是这个类的实例。
紧接着就会执行到第三步创建数据库连接对象实例,接下里将会对这一步进行解析。
紧接着会来到文件实际执行的为 new \think\db\Query,最终会返回执行查询 返回数据集,返回数据为返回 object(think\db\Query)
关于这个$this->connection是在本类的构造函数进行设置的。
先简单的看一下这个构造函数,在这个构造函数中直接就设置了connection这个属性的值,所以在上图中可以使用。
在这里执行完成之后就会将返回的值给从一开始就解析的这个调用未声明的静态方法会进行调用。
其中static::connect()就是最终返回的值static::connect() 返回 object(think\db\Query)。
所以接下来代码会执行到 thinkphp/library/think/db/Query.php 的 query方法
$sql 就是在Db::query()中传递的sql语句,并且执行查询 返回数据集
最后这段代码会执行think\db\connector\Mysql的query方法
接下来来到think\db\connector\Mysql的query方法
在这个方法中主要做了三件事情。
$this->initConnect 初始化数据库连接
$this->PDOStatement->execute(); 执行查询
return $this->getResult($pdo, $procedure); 返回结果集
解析$this->initConnect 初始化数据库连接
在这个方法中可以看到是进行了一次配置信息获取,首先需要明白这个配置信息是什么。
这个配置项是在配置文件database中配置的,根据注释提供的信息可以看到主要是关于主从服务器设置的。
一般情况下是不会在框架中配置主从信息的,这里就不去解析框架是如何实现数据库的主从配置了。
在这个判断中在进行了一次判断当前数据库连接的id,然后执行了连接数据库方法。
这个方法最终会返回object(PDO)#33的一个实例信息。
$this->PDOStatement->execute(); 执行查询
第二件事情做的就是执行查询,接下来我们来详细说明一下这个到底是如何执行的。
在返回pdo实例时,将这个实例赋值给了$this->PDOStatement这个属性,所以会去PDO类中进行执行。
在这里大家需要明白一件事情就是关于execute这个方法,用于执行返回多个结果集、多个更新计数或二者组合的语句。
第三件事情返回结果集
直到这里就是执行的最后一步就是返回结果集。
这里使用的方法都是查询底层,就在去解析了,在这里就会返回最终查询结果。
最终结果就会返回给这里__callStatic方法,并且返回给上层的$res变量。
直到这里关于使用Db查询的执行流程就解析完了, 但是框架给封装的方法不仅仅只有query,其它的查询方式可以按照咔咔的这个流程在进行简单的分析。
最后执行的都会是这一节的最后几个流程,只是前边执行会有一点点区别而已。