PHP 若Mysql单表存在数据量过大,进行逻辑分表

javascript/jquery

浏览数:183

2020-6-26

假设数据库存在一个log表,用于记录用户操作系统的使用记录。
有如下字段
user_id 操作人id
record_time 操作时间
operate_type 操作类型
record_content 操作内容

这个单表情况下,若日积月累的情况下,数据会逐日增多,最终达到百万行的情况,这个时候如果是查询表数据,会很慢。
于是,在PHP开发中可以在插入数据之前进行逻辑处理,并将单表分割成多个表。

原理

以日志表做示例。
根据当前时间的年月做基础,使用PHP动态创建表,达到在2020-03月存储的数据在log_2020_03表中,而2020-04月存储的日志在log_2020_04表中

PHP代码

<?php
/**
 * 日志记录类
 */
class Log
{
    /**
     * 单例模式,防止程序多个地方进行class log的调动,而造成类占用资源重复
     */
    private static $_instance;
    private function __construct() {}
    private function __clone() {}
    public static function getInstance() 
    {
        if (!(self::$_instance instanceof self)) {
            self::$_instance = new self;
        }
        return self::$_instance;
    }
    /**
     * 调用接口
     */
    public function record($userid,$typeId,$content,$time)
    {
        $redis = new Redis();
        $redis->connect( '127.0.0.1', '6379' );    //redis服务器地址和端口
        $redis->auth( 'password' );                //redis密码
        $redis->select( 'db' );                    //选择redis库

        //log表名按照按照 'log_年_月'的规则
        $ruleTable = 'log_' . date('Y-m');

        $isExist = $redis->exists($ruleTable);    //检测redis中是否有log表的标志
        //若redis查询时未发现log表的标志,则动态创建log表
        if ( !$isExist ) {
            $sql = "CREATE TABLE `{$ruleTable}` (
                 `user_id` int(10) NOT NULL,
                 `record_time` datetime NOT NULL,
                 `operate_type` tinyint(10) NOT NULL,
                 `record_content` varchar(20) COLLATE utf8_unicode_ci NOT NULL
                ) ENGINE=MyISAM DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci";
            //执行mysql 此处省略数据库连接代码
            $result = mysqli_query($sql);
            //并记录log表标志到redis中
            $redis->setex($ruleTable,-1,1);        //redis key位'log_年_月' 永久不过期 值为1
        }
        /**
         * 插入记录数据
         */
        $sql = "INSERT INTO `log` (`user_id`, `record_time`, `operate_type`, `record_content`) 
                VALUES ('{$userId}', '{$time}, '{$typeId}', '{$content}')";
        //执行mysql 此处省略数据库连接代码
        $result = mysqli_query($sql);
        //关闭redis连接
        $redis->close();
    }
}
Log::getInstance()->record( 1,10,'随便记录',date('Y-m-d H:i:s') );

作者:如若时光萧瑟去丶