我们公司使用的php框架是Yii2,最近因为线上出现了数据库连接被打满,导致线上业务出现异常的情况,就专门研究了下Yii的数据库链接问题。
1、Yii2怎么链接数据库的
我们项目中使用的是主从分离的mysql数据库。使用的是配置中的components
的方式,其中每个数据库的配置都是一个component
。这样就将数据库的配置注册到Yii::$app中,可以在model中直接在getDb
中获取数据库的配置信息,如下
$config = [
"components" => [
"exam" => [
"class" => "yii\db\Connection", //使用的数据库连接类
"charset" => "utf8", //使用的连接编码方式
"enableSchemaCache" => true, //启用数据库schema的缓存
"schemaCacheDuration" => 3600, //缓存时间1小时
"username" => "", //用户名
"dsn" => "",
"masterConfig" => [...], //主库配置
"masters" => [], //主库dsn列表
"slaveConfig" => [...], //从库配置,
"slaves" => [], //从库dsn配置
]
]
];
这样我们就配置了一个exam
的数据库,而在实际的model中时,可以在这个数据库的基类中主从使用的数据库配置:
class Exam extends \yii\db\ActiveRecord {
public static function getDb() {
return \Yii::$app->exam;
}
}
数据库中每个表都有一个自己的model类,继承这个基类,这样在数据库表中执行的数据库操作就会自动的使用这个数据库的配置。
同时,在进行数据库操作时,框架会自动的进行数据库的读写分离,插入、更新和删除的操作会使用主库,查询操作会自动使用从库。当需要在主库查询时,也可以使用getDb()->useMaster()
来从主库进行查询。
在获取从库连接的时候,如果获取从库连接失败,并且允许获取主库(默认是允许自动调整的),就会自动获取主库的连接。
2、Yii2什么时候建立连接
2.1 数据库什么时候建立连接呢?有两种可能:
1、在请求到来之后,Yii2进行初始化的时候建立连接。
这个是不可能的,我们的项目配置了几十个数据库,而每次请求来了的时候不可能为每个数据库建立一个连接,这是一种浪费:一方面一个请求不可能用到所有的数据库;另一方面建立了数据库连接又不使用,对数据库的资源也是一种浪费。
2、在执行数据库model方法的时候
这样只有在使用到相应数据库查询的时候,才会建立该数据库的连接,这样可以尽可能的优化数据库资源的使用。
同时在一个请求中,如果已经建立了一个数据库的连接,接下来执行类似操作时就会复用这个数据库的连接,减少数据库链接的建立。
2.2 建立数据库连接的方法
在建立数据库连接的时候,最底层调用的方法时Yii2\db\Conection
类中的openFromPool
方法,从方法名看,是从池子中打开一个连接,其实这里是从之前配置的数据库列表中随机获取一个连接:
protected function openFromPool(array $pool, array $sharedConfig) {
if (empty($pool)) {
return null;
}
if (!isset($sharedConfig['class'])) {
$sharedConfig['class'] = get_class($this);
}
$cache = is_string($this->serverStatusCache) ? Yii::$app->get($this->serverStatusCache, false) : $this->serverStatusCache;
//这里会先将从库的地址列表打乱,然后循环这个配置列表,逐一建立连接,直到建立连接成功。如果都建立失败会返回null
shuffle($pool);
foreach ($pool as $config) {
...
try {
//这里就是实际的建立连接的方法
$db->open();
return $db;
}
}
return null;
这里有一个要注意的点,在调用openFromPool
建立连接的时候,底层使用的实际方法是open
。这里要注意,在获取从库时,是使用生成的$db
对象来建立的,而获取主库时直接使用的$this
,所以在open方法中会判断是否有$this->masters
,只有在主库时才会有值,才会建立主库的连接。
这里最开始以为时每次都建立主库连接,直到后来看到$db
才发现不是最开始的Connection对象了。
3、Yii2什么时候销毁链接
上面说了,我们是在执行数据库表对应的model文件的方法时,才会建立数据库连接,而这个数据库的连接会被放到db对象中,其中从库会放到_slave
里面,主库的连接放到pdo
里面。这样在获取的时候,如果有值,就会直接使用。
具体的销毁是在php请求完毕之后,当php把返回值发送给nginx之后,php释放相关的对象值时。
最后,使用Yii2时还是推荐使用主从数据库配置的,这样可以减少主服务器的压力,避免造成主服务器的问题,同时,如果查询压力太大的时候,可以很方便的扩展从库,增加系统的抗压能力。