-一个简单的数据缓存解决方案
admin
2023-06-22 18:02:35
0

    最近做了一个聊天室系统,系统有N个聊天室,在并发量很大的情况下,如果用户聊天记录直接写入MySql的话,对数据库的冲击会很大。这里我写了个简单的解决方案:聊天数据先存入redis的队列中,然后用crontab每隔1分钟执行PHP脚本,把数据从redis队列中“批量”移到mysql。当有多条插入语句时,把数据拼装成1insert语句,减少数据库的操作次数。

 

1. 配置文件 config.php

function dbMessageConfig(){

    //用户信息库表的配置

return array(

        'database_type' => 'mysql',

        'database_name' => 'test',

        'server' => '127.0.0.1',

        'username' => 'root',

        'password' => 'root',

        'charset' => 'utf8'

);

}

 

function redisCacheConfig(){

return array(

        'ip' => '127.0.0.1',

        'port' => '6379',

        'password' => '888888'

);

}

 

2. redis2mysql.php

error_reporting(E_ALL^E_NOTICE^E_WARNING);

 

require_once dirname(__FILE__).'/config.php';

 

$dbMessage = dbMessageConfig();

$configRedis = redisCacheConfig();

 

$r = new \Redis();

$r->connect($configRedis['ip'], $configRedis['port']);

$r->auth($configRedis['password']);

 

//获取redis中的消息

$message = $r->rpop('chat_message');

$tenantArr = array();

if(!empty($message)){

    $tmpArr = explode(':', $message);

    $tenant_id = $tmpArr[0];

    if(!empty($tenant_id) && is_numeric($tenant_id)){

    if(!empty($tmpArr[1])) $tenantArr[$tenant_id] = $tmpArr[1];

}

}

 

//一次最多取20条记录

$MAX_NUM = 20;

$index = 1;

while(!empty($message) && $MAX_NUM > $index){

    $message = $r->rpop('chat_message');

    //echo $message;

    $tmpArr = explode(':', $message);

    $tenant_id = $tmpArr[0];

    if(!empty($tenant_id) && is_numeric($tenant_id)){

    if(!empty($tmpArr[1])){

    if(array_key_exists($tenant_id, $tenantArr)){

    $tmp = $tenantArr[$tenant_id];

$tmp = $tmp.",".$tmpArr[1];

$tenantArr[$tenant_id] = $tmp;

}else{

    $tenantArr[$tenant_id] = $tmpArr[1];

}

}

}


$index = $index + 1;

}

 

$table_message = 'chat_message';

 

$dbh = new \PDO('mysql:host='.$dbMessage['server'].';dbname='.$dbMessage['database_name'], $dbMessage['username'], $dbMessage['password']);

$dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);    

$dbh->exec('set names '.$dbMessage['charset']);

 

foreach ($tenantArr as $tenant_id=>$sql) {

$sqlstr = 'insert INTO '.$table_message.'_'.$tenant_id.' VALUES '.$sql.';';


    $stmt = $dbh->prepare($sqlstr);  

    $stmt->execute();

}

 

3. 配置crontab

*/1 * * * * /opt/php/bin/php -f /usr/local/php/redis2mysql.php >/dev/null 2>&1

 

启动crontab/sbin/service crond start

 

4. 写入redissql语句(java代码):

//把chatmessage发送到redis储存

Jedis redis = RedisUtil.getJedis();

String key="message";

StringBuffer strMsg = new StringBuffer(tenant_id);

strMsg.append(":(NULL,");

strMsg.append(room_id);

strMsg.append(",");

strMsg.append(uid);

strMsg.append(",'");

strMsg.append(username);

strMsg.append("',");

strMsg.append(user_type);

strMsg.append(",'");

strMsg.append(avatar);

strMsg.append("','");

strMsg.append(content);

strMsg.append("',");

strMsg.append(add_time);

strMsg.append(",1)");


redis.lpush(key, strMsg.toString());

 

参考:

 crontab启动 、运行和编辑查看

http://blog.csdn.net/ariessurfer/article/details/7459183

 

 crontab定时任务配置

http://www.cnblogs.com/kerrycode/p/3238346.html

 

 php mysql PDO使用

http://blog.csdn.net/qq635785620/article/details/11284591

相关内容

热门资讯

凤凰晚报丨美国牡蛎养殖场,走出... 今日人物【美国牡蛎养殖场,走出一位特朗普“平替”】2026年5月,在美国缅因州波特兰举行的一场州民主...
胡锡进:记者暗访被打,我有这样... 一名女记者暗访助孕实验室遭粗暴对待,被拖拽骨折。警方已对这起故意伤害刑事立案调查。支持记协对该记者的...
福州一立体停车场闲置长草?当地... 澎湃新闻记者 王选辉在福建省福州市仓山区的金山文体中心旁,有一座立体停车库。近日有周边群众向澎湃新闻...
《微博AI内容生态报告》:金橙... 2026年5月28日,《微博AI内容生态报告》(以下简称“报告”)正式发布。本报告从微博平台视角出发...
把超级AI助理“戴”进日常 讯... (澳门,2026年5月28日)当下,人工智能正以全新软硬件形态重塑人机交互方式。5月28日,科大讯飞...
2026年演唱会拍清舞台手机推... 对于演唱会爱好者而言,最令人沮丧的莫过于花费重金抢到门票,却因为座位偏远,手机长焦无力,拍出的舞台画...
联通+联想携手 首款AI+eS... 2026年5月28日,中国联通与联想集团联合举办拯救者Y70新一代“AI+eSIM”游戏手机战略合作...
国家大基金领投DeepSeek... DeepSeek的模型迭代需要持续投入大规模训练算力,充足资金是支撑这一长周期投入的重要来源 文|《...
“武器系统成熟前不准造”,美国... 【文/观察者网 山猫】尽管最近传出了多个关于“特朗普”级战列舰项目似乎确实要开始着手实施的迹象,包括...
90后小伙徒步重走长征路,当事... “你相信人可以重生吗?肉体不可以,但我相信精神可以。”来自新疆的90后小伙王成于2025年10月10...