假设数据库存在一个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') );
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。