ThinkChat2.0新版上线,更智能更精彩,支持会话、画图、阅读、搜索等,送10W Token,即刻开启你的AI之旅 广告
##2.12.1 背景 为了应对产品海量用户的愿景需求,这里将设计一个分布式的数据库存储方案,以便能满足数据量的骤增、云服务的横向扩展、后台接口开发的兼容性,以及数据迁移等问题,避免日后因为全部数据都存放在单台服务器上的限制。 ##2.12.2 主要思想 + 1、分库分表 + 2、路由规则 + 3、扩展字段 + 4、可配置 + 5、SQL语句自动生成 ###(1)分库分表 是指将不需要进行必要关联查询的表分开存放,如存放事件推送的weili_event_pushto和存放标签的weili_tag;同时,对于 同一个表,因为存放的数据量是可预见式的暴增,如上述的weili_event_pushto,每时每刻都会产生大量的来自用户发布的事件,因此为了突破 MySQL单表的限制以及其他问题,需要将此表同时创建N份。 ###(2)路由规则 在上面进行了分库分表后,开发人员在读取时,就需要根据相应的规则找到对应 的数据库和数据库表,这里建议每个表都需要有int(11)类型的id字段,以便作为分表的参考。 ###(3)扩展字段 在完成了分库分表和制定路由规则后,考虑到日后有数据库的DB变更,为减少DB变更对现有数据库表的影响,这里建议每个表都增加text类型的extra_data字段,并且使用json格式进行转换存储。 ###(4)可配置 在有了N台数据库服务器以及每个表都拆分成M张表后,为减少后台接口开发人员的压力,有必须在后台接口框架提供可配置 的支持。即:数据库的变更不应影响开发人员现有的开发,也不需要开发人员作出代码层面的改动,只需要稍微配置一下即可。关于这块,请见下面的框架实现部 分。 ###(5)SQL语句自动生成 对于相同表的建表语句,可以通过脚本来自动生成,然后直接导入数据即可。 ##2.12.3 PhalApi框架的实现方案 PhalApi框架主要需要实现的是路由这一层的映射,并且通过可配置的方式进行控制,同时还应支持生产环境和测试环境的异同,如在测试环境我们明显不需要1000张数据库的表。为此,需要提供一种 **表名 + id** 映射到 **数据库服务器 + 具体哪张表** 的规则。 ![show](http://webtools.qiniudn.com/20150411005257_8ea80ab617023e2f377bf1ba8fdff6c5) 如上图所示,表名会统一加上前缀,并且将id按一定的表总数进行取模,最后再根据得到的具体表名,通过映射表查找到对应 的数据库服务器进行操作。其中,model层为开发实现,数据库表的映射由接口框架实现支持。 ##2.12.4 使用示例 ###(1)配置数据库的路由配置 修改./Config/dbs.php文件,以下是参考的示例配置。其中servers为DB服务器,包括数据库的账号信息等,tables为数据库表的映射关系,其中__default__下标为缺省的数据库路由。 在每个数据库表里面,可以配置多个数据库表,通过开始的下标start和结束的下标end来对表进行分布式存放,并且如果没有start和end的,则视为不需要拆分存放,同时也是当找不到合适时的拆分表时所采用的默认配置。 ```javascript return array( /** * avaiable db servers */ 'servers' => array( 'db_demo' => array( 'host' => 'localhost', //数据库域名 'name' => 'test', //数据库名字 'user' => 'root', //数据库用户名 'password' => '123456', //数据库密码 'port' => '3306', //数据库端口 ), ), /** * custom table map */ 'tables' => array( '__default__' => array( 'prefix' => 'tbl_', 'key' => 'id', 'map' => array( array('db' => 'db_demo'), ), ), 'demo' => array( 'prefix' => 'tbl_', 'key' => 'id', 'map' => array( array('db' => 'db_demo'), array('start' => 0, 'end' => 2, 'db' => 'db_demo'), ), ), ), ); ``` 上面示例配置的意思是: ```javascript 表名 DB服务器 tbl_demo db_demo tbl_demo_0 db_demo tbl_demo_1 db_demo tbl_demo_2 db_demo ``` ###(2)准备需要创建表的基本SQL语句 这里说的基本SQL语句是指:仅是这个表所特有的字段,排除已固定公共有的自增主键id,和扩展字段ext_data。下面是一个示例: ```javascript `name` varchar(11) DEFAULT NULL, ``` ###(3)生成并导入SQL语句 由于拆分后的数据库表数量众多,这里提供了一个快捷的脚本工具来生成所需要创建的数据库表。 ```javascript $ php ./build_sqls.php Usage: ./build_sqls.php <table> [engine=InnoDB] ``` 执行上面的脚本,输入数据库表参数后: ```javascript php ./build_sqls.php demo ``` 将会从配置文件 里面寻找所需要创建的表,并生成类似以下的SQL语句: ```javascript /** * DB: localhost db_demo */ CREATE TABLE `demo` ( `id` int(11) unsigned NOT NULL AUTO_INCREMENT, `name` varchar(11) DEFAULT NULL, `ext_data` text COMMENT 'json data here', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8; /** * DB: localhost db_demo */ CREATE TABLE `tpl_demo_0` ( `id` int(11) unsigned NOT NULL AUTO_INCREMENT, `name` varchar(11) DEFAULT NULL, `ext_data` text COMMENT 'json data here', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8; CREATE TABLE `tpl_demo_1` ( `id` int(11) unsigned NOT NULL AUTO_INCREMENT, `name` varchar(11) DEFAULT NULL, `ext_data` text COMMENT 'json data here', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8; CREATE TABLE `tpl_demo_2` ( `id` int(11) unsigned NOT NULL AUTO_INCREMENT, `name` varchar(11) DEFAULT NULL, `ext_data` text COMMENT 'json data here', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8; ``` ###(4)使用与代码开发 在将上面的SQL语句导入数据库后,即可以像之前那样操作数据库。下面是一些示例: ```javascript DI()->notorm = new PhalApi_DB_NotORM(PhalApi_DI::one()->config->get('dbs'), true); DI()->notorm->demo->where('id', '1')->fetch(); ``` 用到了拆分表的代码示例,假设event表被拆分成了3个表,则客户端在调用里,需要根据(id % 3 )来拼接合适的数据库表名,其他使用不变。 ```javascript DI()->notorm = new PhalApi_DB_NotORM(PhalApi_DI::one()->config->get('dbs'), true); $row = DI()->notorm->demo_0->where('id', '3')->fetch(); $row = DI()->notorm->demo_1->where('id', '10')->fetch(); $row = DI()->notorm->demo_2->where('id', '2')->fetch(); ``` ####使用Model基类的情况 更好的写法,应该是继承于PhalApi_Model_NotORM,并统一实现分表的操作,如: ```javascript <?php class Model_Demo extends PhalApi_Model_NotORM { protected function getTableName($id) { $tableName = 'demo'; if ($id !== null) { $tableName .= '_' . ($id % 3); } return $tableName; } } ``` 然后,上面的查询分别对应: ```javascript $model = new Model_Demo(); $row = $model->get('3', 'id'); $row = $model->get('10', 'id'); $row = $model->get('2', 'id'); ``` 更进一步,我们可以通过$this->getORM($id)来获取分表的实例进行分表的操作,如: ```javascript <?php class Model_Demo extends PhalApi_Model_NotORM { //... ... public function getNameById($id) { $row = $this->getORM($id)->select('name')->fetchRow(); //假设$id为3,则 $this->getORM($id) 等效于 DI()->notorm->demo_0 return !empty($row) ? $row['name'] : ''; } } ``` ##2.12.5 多个数据库的配置方式 当需要使用多个数据库时,可以先在servers中可以配置多组数据库的信息,然后在tables为不同的数据库表指定不同的数据库服务器。 假设我们有两台数据库服务器,分别叫做db_A、db_B,即: ```javascript return array( /** * DB数据库服务器集群 */ 'servers' => array( 'db_A' => array( //db_A 'host' => '192.168.0.1', //数据库域名 // ... ... ), 'db_B' => array( //db_B 'host' => '192.168.0.2', //数据库域名 // ... ... ), ), //... ... ``` 若db_A服务器中的数据库有表a_table_user、a_table_friends,而db_B服务器中的数据库有表b_table_article、b_table_comments,则: ```javascript <?php return array( //... ... /** * 自定义路由表 */ 'tables' => array( //通用路由 '__default__' => array( 'prefix' => 'a_', //以 a_ 为表前缀 'key' => 'id', 'map' => array( array('db' => 'db_A'), //默认,使用db_A数据库 ), ), 'table_article' => array( //表b_table_article 'prefix' => 'b_', //表名前缀 'key' => 'id', //表主键名 'map' => array( //表路由配置 array('db' => 'db_B'), // b_table_article表使用db_B数据库 ), ), 'table_comments' => array( //表b_table_article 'prefix' => 'b_', //表名前缀 'key' => 'id', //表主键名 'map' => array( //表路由配置 array('db' => 'db_B'), // b_table_comments表使用db_B数据库 ), ), ), ``` 如果项目存在分表的情况,可结合上述的分表的说明进行配置。 这里为了让大家更为明了,假设db_A服务器中的数据库有表a_table_user、a_table_friends_0到a_table_friends_9(共10张表), 而db_B服务器中的数据库有表b_table_article、b_table_comments_0到b_table_comments_19(共20张表),则结合起来的完整配置为: ```javascript <?php return array( /** * DB数据库服务器集群 */ 'servers' => array( 'db_A' => array( //db_A 'host' => '192.168.0.1', //数据库域名 // ... ... ), 'db_B' => array( //db_B 'host' => '192.168.0.2', //数据库域名 // ... ... ), ), /** * 自定义路由表 */ 'tables' => array( //通用路由 '__default__' => array( 'prefix' => 'a_', //以 a_ 为表前缀 'key' => 'id', 'map' => array( array('db' => 'db_A'), //默认,使用db_A数据库 ), ), 'table_friends' => array( //分表配置 'prefix' => 'a_', //表名前缀 'key' => 'id', //表主键名 'map' => array( //表路由配置 array('db' => 'db_A'), // b_table_comments表使用db_B数据库 array('start' => 0, 'end' => 9, 'db' => 'db_A'), //分表配置(共10张表) ), ), 'table_article' => array( //表b_table_article 'prefix' => 'b_', //表名前缀 'key' => 'id', //表主键名 'map' => array( //表路由配置 array('db' => 'db_B'), // b_table_article表使用db_B数据库 ), ), 'table_comments' => array( //表b_table_article 'prefix' => 'b_', //表名前缀 'key' => 'id', //表主键名 'map' => array( //表路由配置 array('db' => 'db_B'), // b_table_comments表使用db_B数据库 array('start' => 0, 'end' => 19, 'db' => 'db_B'), //分表配置(共20张表) ), ), ), ); ``` ##2.12.6 与主从数据库的有机结合 虽然这是专门为海量数据设计的存储方案,但也是可以结合主从配置来获得更庞大强壮的方案,当然为之付出的是复杂性的引入。 简单地,可以将dbs.php复制一份dbs_slave.php出来给从库使用,然后注册一个从库的服务: ```javascript DI()->slaveNotorm = new PhalApi_DB_NotORM(DI()->config->get('slave_dbs')); ``` 最后,在需要使用从库来读取时,使用slaveNotorm 服务即可。 ##2.12.7 不足与注意点 这样的设计是有明显的灵活性的,因为在后期如果需要迁移数据库服务器,我们可以在框架支持的情况下轻松应对,但依然需要考虑到一些问题和不足。 ###(1)DB变更 DB变更,这块是必不可少的,但一旦数据库表被拆分后,表数量的骤增导致变更执行困难,所以这里暂时使用了一个折中的方案,即提供了一个ext_data 扩展字段用于存放后期可能需要的字段信息,建议采用json格式,因为通用且长度比序列化的短。但各开发可以根据自己的需要决定格式。即使如此,扩展字段 明显做不到一些SQL的查询及其他操作。 ###(2)表之间的关联查询 表之间的关联查询,这个是分拆后的最大问题。虽然这样的代价是我们可以得到更庞大的存储设计, 而且很多表之间不需要必须的关联的查询,即使我们需要,我们也可以通过其他手段如缓存和分开查询来实现。这对开发人员有一定的约束,但是对于可预见性的海 量数量,这又是必须的。