Redis 缓存保存某段时间累加的数值,加入最大id防止同一秒并发过大,导致只统计了执行时同一秒的部分数据,而同一秒另一部分数据在下次累加时没有统计到缓存中

   //coin总数
public function get_stat_coin_total($key = '',$now = 0){
$redisTotal = 0;
$redisMaxId = 0;
$redisData = null;
$effective_time = Kv::getStatEffectiveTime(); //有效时间
if($key == 'gift_log_coin_today'){
$time1 = strtotime(date('Y-m-d',strtotime('0 day')));
$time2 = $now; //统计到当前时间的数据
}elseif($key == 'gift_log_coin_yesterday'){
$time1 = strtotime(date('Y-m-d',strtotime('-1 day')));
$time2 = strtotime(date('Y-m-d',strtotime('0 day')));
}elseif($key == 'gift_log_coin_yesterday_before'){
$time1 = 0; //最开始时间
$time2 = strtotime(date('Y-m-d',strtotime('0 day')));
}elseif($key == 'gift_log_coin_total'){
$time1 = 0; //最开始时间
$time2 = $now; //统计到当前时间的数据
}else{
return 0;
}
$redis = self::getRedis();
$userStatListKey = UserStat::USER_STAT_LIST_KEY . $key;
$redisData = $redis->hGetAll($userStatListKey);
if($redisData && $redisData['effective_time'] > $now){
$time1 = $redisData['end_time'];
$redisTotal = $redisData['total'];
$redisMaxId = $redisData['max_id'];
$effective_time = $redisData['effective_time'];
} $sql = "SELECT sum(send_coin) as total,max(id) as max_id FROM `gift_log` WHERE add_time >= {$time1} AND add_time <= {$time2} AND id > {$redisMaxId}; ";
$connection = $this->getReadConnection();
$data = $connection->fetchOne($sql);
$total = empty($data['total'])?0:$data['total'];
$max_id = empty($data['max_id'])?$redisMaxId:$data['max_id'];
$result = [
'start_time'=>$time1,
'end_time'=>$time2,
'total'=>$total+$redisTotal,
'max_id'=>$max_id,
'effective_time'=>$effective_time,
];
$redis->hMset($userStatListKey,$result); return $result['total']; }

一条语句判断数据是否异常例子(保存统计过的最大自增id)

/**
* 检测 今日 coin 是否正常 统一查询
* 保存统计过的最大自增id
*/
public function checkCoin() {
$model = new UserStat();
$connection = $model->getReadConnection();
$redis = self::getRedis();
$userStatListKey = UserStat::USER_STAT_LIST_KEY;
$redisData = $redis->hGetAll($userStatListKey);
if ($redisData) {
$sql = <<<SQL
SELECT (SELECT concat(IFNULL(sum(coin),0),'_',ifnull(max(id),:max_recharge_id)) FROM wallet_recharge_log WHERE pay_status = 1 AND id > :max_recharge_id ) AS recharge
,(SELECT concat(IFNULL(sum(-send_coin),0),'_',ifnull(max(id),:max_send_gift_id)) AS send_total FROM gift_log WHERE id > :max_send_gift_id) AS send_gift
,(SELECT concat(IFNULL(sum(CASE type WHEN 1 THEN coin ELSE -coin END),0),'_',ifnull(max(id),:max_wallet_order_id)) FROM wallet_order_log WHERE id > :max_wallet_order_id ) AS wallet_order
,(SELECT IFNULL(sum(-coin),0) AS has_total_coin FROM user_wallet) AS has_coin
,(SELECT IFNULL(sum(-pay_coin),0) AS total FROM live_pay_log WHERE `status` =2) AS live_pay
,(SELECT concat(IFNULL(sum(-coin),0),'_',ifnull(max(id),:max_barrage_pay_id)) FROM user_barrage_log WHERE id > :max_barrage_pay_id) AS barrage_pay
,(SELECT concat(IFNULL(sum(-coin),0),'_',ifnull(max(id),:max_vip_pay_id)) FROM user_vip_log WHERE id > :max_vip_pay_id) AS vip_pay
,(SELECT concat(IFNULL(sum(coin),0),'_',ifnull(max(id),:max_exchange_get_id)) FROM wallet_exchange_log WHERE id > :max_exchange_get_id) AS exchange_get;
SQL;
$bind_arr = [
'max_recharge_id' => $redisData['max_recharge_id'],
'max_send_gift_id' => $redisData['max_send_gift_id'],
'max_wallet_order_id' => $redisData['max_wallet_order_id'],
'max_barrage_pay_id' => $redisData['max_barrage_pay_id'],
'max_vip_pay_id' => $redisData['max_vip_pay_id'],
'max_exchange_get_id' => $redisData['max_exchange_get_id'],
];
$result = $connection->query($sql, $bind_arr)->fetch(); $change_coin = $redisData['has_coin'] - $result['has_coin']; $result['recharge'] = explode('_',$result['recharge']);
$result['send_gift'] = explode('_',$result['send_gift']);
$result['wallet_order'] = explode('_',$result['wallet_order']);
$result['barrage_pay'] = explode('_',$result['barrage_pay']);
$result['vip_pay'] = explode('_',$result['vip_pay']);
$result['exchange_get'] = explode('_',$result['exchange_get']);
$redis_arr = [
'has_coin' => $result['has_coin'],
'live_pay' => $result['live_pay'],
'recharge' => $redisData['recharge'] + $result['recharge'][0],
'send_gift' => $redisData['send_gift'] + $result['send_gift'][0],
'wallet_order' => $redisData['wallet_order'] + $result['wallet_order'][0],
'barrage_pay' => $redisData['barrage_pay'] + $result['barrage_pay'][0],
'vip_pay' => $redisData['vip_pay'] + $result['vip_pay'][0],
'exchange_get' => $redisData['exchange_get'] + $result['exchange_get'][0],
];
$max_id_arr = [
'max_recharge_id' => $result['recharge'][1],
'max_send_gift_id' => $result['send_gift'][1],
'max_wallet_order_id' => $result['wallet_order'][1],
'max_barrage_pay_id' => $result['barrage_pay'][1],
'max_vip_pay_id' => $result['vip_pay'][1],
'max_exchange_get_id' => $result['exchange_get'][1],
];
echo "send_gift : [{$result['send_gift'][0]}];change_coin [$change_coin]']\n\n";
} else {
$sql = <<<SQL
SELECT (SELECT concat(IFNULL(sum(coin),0),'_',ifnull(max(id),0)) FROM wallet_recharge_log WHERE pay_status = 1 ) AS recharge
,(SELECT concat(IFNULL(sum(-send_coin),0),'_',ifnull(max(id),0)) AS send_total FROM gift_log) AS send_gift
,(SELECT concat(IFNULL(sum(CASE type WHEN 1 THEN coin ELSE -coin END),0),'_',ifnull(max(id),0)) FROM wallet_order_log ) AS wallet_order
,(SELECT IFNULL(sum(-coin),0) AS has_total_coin FROM user_wallet) AS has_coin
,(SELECT IFNULL(sum(-pay_coin),0) AS total FROM live_pay_log WHERE `status` =2) AS live_pay
,(SELECT concat(IFNULL(sum(-coin),0),'_',ifnull(max(id),0)) FROM user_barrage_log ) AS barrage_pay
,(SELECT concat(IFNULL(sum(-coin),0),'_',ifnull(max(id),0)) FROM user_vip_log) AS vip_pay
,(SELECT concat(IFNULL(sum(coin),0),'_',ifnull(max(id),0)) FROM wallet_exchange_log) AS exchange_get;
SQL;
$result = $connection->query($sql)->fetch();
$result['recharge'] = explode('_',$result['recharge']);
$result['send_gift'] = explode('_',$result['send_gift']);
$result['wallet_order'] = explode('_',$result['wallet_order']);
$result['barrage_pay'] = explode('_',$result['barrage_pay']);
$result['vip_pay'] = explode('_',$result['vip_pay']);
$result['exchange_get'] = explode('_',$result['exchange_get']);
$redis_arr = [
'has_coin' => $result['has_coin'],
'live_pay' => $result['live_pay'],
'recharge' => $result['recharge'][0],
'send_gift' => $result['send_gift'][0],
'wallet_order' => $result['wallet_order'][0],
'barrage_pay' => $result['barrage_pay'][0],
'vip_pay' => $result['vip_pay'][0],
'exchange_get' => $result['exchange_get'][0],
];
$max_id_arr = [
'max_recharge_id' => $result['recharge'][1],
'max_send_gift_id' => $result['send_gift'][1],
'max_wallet_order_id' => $result['wallet_order'][1],
'max_barrage_pay_id' => $result['barrage_pay'][1],
'max_vip_pay_id' => $result['vip_pay'][1],
'max_exchange_get_id' => $result['exchange_get'][1],
];
}
$coin_diff_today = array_sum($redis_arr);
$coin_limit = Kv::get('coin_limit', '0');
//coin是否正常
if (abs($coin_diff_today) < $coin_limit) {
//正常
$redis_arr['status'] = 1;
} else {
//异常
$redis_arr['status'] = 2;
}
$redis_arr = array_merge($redis_arr,$max_id_arr);
$redis_arr['coin_diff_today'] = $coin_diff_today;
$redis->hMset($userStatListKey,$redis_arr);
if(!$redisData){
$redis->expire($userStatListKey,3600 * 24);
}
return $redis_arr;
}

redis 数据统计(用自增id防止同一秒并发过大没统计成功)的更多相关文章

  1. mysql插入数据后返回自增ID的方法,last_insert_id(),selectkey

    mysql插入数据后返回自增ID的方法 mysql和oracle插入的时候有一个很大的区别是,oracle支持序列做id,mysql本身有一个列可以做自增长字段,mysql在插入一条数据后,如何能获得 ...

  2. mysql函数之六:mysql插入数据后返回自增ID的方法,last_insert_id(),selectkey

    mysql插入数据后返回自增ID的方法 mysql和oracle插入的时候有一个很大的区别是,oracle支持序列做id,mysql本身有一个列可以做自增长字段,mysql在插入一条数据后,如何能获得 ...

  3. mysql清空表数据并重置自增ID

    mysql清空表数据并重置自增ID: ## 查看mysql> select * from work_order_company;mysql> show create table work_ ...

  4. mysql插入数据后返回自增ID的方法

    mysql和oracle插入的时候有一个很大的区别是,oracle支持序列做id,mysql本身有一个列可以做自增长字段,mysql在插入一条数据后,如何能获得到这个自增id的值呢? 方法一是使用la ...

  5. mybatis插入数据并返回自增Id

    上图mybatis的写法,在xxxMapper.xml中: 加入:useGeneratedKeys="true" keyProperty="applyId" k ...

  6. Mybatis使用generatedKey在插入数据时返回自增id始终为1,自增id实际返回到原对象当中的问题排查

    今天在使用数据库的时候,遇到一个场景,即在插入数据完成后需要返回此数据对应的自增主键id,但是在使用Mybatis中的generatedKey且确认各项配置均正确无误的情况下,每次插入成功后,返回的都 ...

  7. 如何在MySQl数据库中给已有的数据表添加自增ID?

    由于使用MySQL数据库还没有多久的缘故,在搭建后台往数据库导入数据的时候发现新增的表单是没有自增id的,因次就有了上面这个问题. 解决方法 1.给某一张表先增加一个字段,这里我们就以node_tab ...

  8. Sql Server插入数据并返回自增ID,@@IDENTITY,SCOPE_IDENTITY和IDENT_CURRENT的区别

    预备知识:SQLServer的IDENTITY关键字IDENTITY关键字代表的是一个函数,而不是identity属性.在access里边没有这个函数,所以在access不能用这个语句.语法:iden ...

  9. Sql Server插入数据并返回自增ID,@@IDENTITY,SCOPE_IDENTITY和IDENT_CURRENT的区别(转载)

    预备知识:SQL Server的IDENTITY关键字IDENTITY关键字代表的是一个函数,而不是identity属性.在access里边没有这个函数,所以在access不能用这个语句.语法:ide ...

随机推荐

  1. vuex中的dispatch和commit

    dispatch:含有异步操作,eg:向后台提交数据,写法: this.$store.dispatch('mutations方法名',值) commit:同步操作,写法:this.$store.com ...

  2. xxx.app已损坏,打不开.你应该将它移到废纸篓-已解决

    到这种情况可能不是你的软件包的问题,可能是macos安全设置问题. 解决办法就是: 1.系统偏好设置... -> 安全性与隐私-->修改为任何来源 2.serria里面没有“任何来源”这一 ...

  3. [xdoj] 1301&1302 数字计数 数字计数的复仇

    1.首先需要掌握二进制数的一种特性,00=0,01=1,10=2,11=3.每一个二进制的值代表他前面的二进数的个数,比如11=3,他的前面就有三个二进制的数字,不过在本题中,题目数据是1-n,故把0 ...

  4. android怎么抓取双向认证https的包

    这里仅提供思路. 第一种方法: dex层面,可以直接用插日志方法,找到app使用的https库,这里以某app为例,使用okhttp, okhttp收发包相关代码: Request request = ...

  5. InstallShield2015制作安装包----------安装过程中修改文件内容

    //修改安装目录下autostart.vbs里的路径 //打开文件 OpenFileMode(FILE_MODE_NORMAL); strPath=INSTALLDIR+"centerAut ...

  6. CSU 1862 The Same Game(模拟)

    The Same Game [题目链接]The Same Game [题目类型]模拟 &题解: 写这种模拟题要看心态啊,还要有足够的时间,必须仔细读题,一定要写一步,就调试一步. 这题我没想到 ...

  7. C# mongodb $set或$addToSet批量更新很慢原因

    C# mongodb $set或$addToSet批量更新很慢原因的解决方法:关键字段要建立索引

  8. websocket发送接收协议

    一.websocket接收数据 1)固定字节(1000 0001或1000 0010);   ---区分是否是数据包的一个固定字节(占1个字节) 个字节是数据的长度; 3)mark 掩码为包长之后的 ...

  9. python 正则基本方法

    2017-04-11 学习python,免不了应对爬虫,初学爬虫最难理解的就是正则表达式. 比如我们要爬去网页上的某些内容,就像下面这种形式: <p>safdsf</p>< ...

  10. VS2013打包程序步骤

    VS自带的打包程序默认是没有安装的,如果有打包的需要,需要自己去下载一个安装程序  1.右击解决方案,选择添加项目,在打开的对话框中找到[已安装]-[模板]-[其他项目]-[安装和部署],如图示.第一 ...