前言

最近在做阅读类的业务,需要记录用户的PV,UV;

项目状况:前期尝试业务阶段;

特点:

  • 快速实现(不需要做太重,满足初期推广运营即可)
  • 快速投入市场去运营

收集用户的原始数据,三要素:

  • 在什么时间
  • 阅读哪篇文章

提到PV,UV脑海中首先浮现特点:

  • 需要考虑性能(每个客户每打开一篇文章进行记录)
  • 允许数据有较小误差(少部分数据丢失)

架构设计

  • 架构图:

  • 时序图
  • 记录基础数据MySQL表结构
CREATE TABLE `zh_article_count` (`id` bigint(20) NOT NULL AUTO_INCREMENT,`bu_no` varchar(32) DEFAULT NULL COMMENT '业务编码',`customer_id` varchar(32) DEFAULT NULL COMMENT '用户编码',`type` int(2) DEFAULT '0' COMMENT '统计类型:0APP内文章阅读',`article_no` varchar(32) DEFAULT NULL COMMENT '文章编码',`read_time` datetime DEFAULT NULL COMMENT '阅读时间',`create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',`update_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '更新时间',`param1` int(2) DEFAULT NULL COMMENT '预留字段1',`param2` int(4) DEFAULT NULL COMMENT '预留字段2',`param3` int(11) DEFAULT NULL COMMENT '预留字段3',`param4` varchar(20) DEFAULT NULL COMMENT '预留字段4',`param5` varchar(32) DEFAULT NULL COMMENT '预留字段5',`param6` varchar(64) DEFAULT NULL COMMENT '预留字段6',PRIMARY KEY (`id`) USING BTREE,UNIQUE KEY `uk_zh_article_count_buno` (`bu_no`),KEY `key_zh_article_count_csign` (`customer_id`),KEY `key_zh_article_count_ano` (`article_no`),KEY `key_zh_article_count_rtime` (`read_time`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='文章阅读统计表';
  • 技术实现方案

    SpringBoot

    Redis

    MySQL

代码实现

  • 完整代码(GitHub,欢迎大家Star,Fork,Watch)

    https://github.com/dangnianchuntian/springboot

  • 主要代码展示

    • Controller
  /** Copyright (c) 2020. zhanghan_java@163.com All Rights Reserved.* 项目名称:Spring Boot实战解决高并发数据入库: Redis 缓存+MySQL 批量入库* 类名称:ArticleCountController.java* 创建人:张晗* 联系方式:zhanghan_java@163.com* 开源地址: https://github.com/dangnianchuntian/springboot* 博客地址: https://zhanghan.blog.csdn.net*/package com.zhanghan.zhredistodb.controller;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;import com.zhanghan.zhredistodb.controller.request.PostArticleViewsRequest;
import com.zhanghan.zhredistodb.service.ArticleCountService;@RestController
public class ArticleCountController {@Autowiredprivate ArticleCountService articleCountService;/*** 记录用户访问记录*/@RequestMapping(value = "/post/article/views", method = RequestMethod.POST)public Object postArticleViews(@RequestBody @Validated PostArticleViewsRequest postArticleViewsRequest) {return articleCountService.postArticleViews(postArticleViewsRequest);}/***  批量将缓存中的数据同步到MySQL(模拟定时任务操作)*/@RequestMapping(value = "/post/batch", method = RequestMethod.POST)public Object postBatch() {return articleCountService.postBatchRedisToDb();}}
  • Service
/** Copyright (c) 2020. zhanghan_java@163.com All Rights Reserved.* 项目名称:Spring Boot实战解决高并发数据入库: Redis 缓存+MySQL 批量入库* 类名称:ArticleCountServiceImpl.java* 创建人:张晗* 联系方式:zhanghan_java@163.com* 开源地址: https://github.com/dangnianchuntian/springboot* 博客地址: https://zhanghan.blog.csdn.net*/package com.zhanghan.zhredistodb.service.impl;import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import java.util.stream.Collectors;import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Service;
import org.springframework.util.CollectionUtils;import com.alibaba.fastjson.JSON;
import com.zhanghan.zhredistodb.controller.request.PostArticleViewsRequest;
import com.zhanghan.zhredistodb.dto.ArticleCountDto;
import com.zhanghan.zhredistodb.mybatis.mapper.XArticleCountMapper;
import com.zhanghan.zhredistodb.service.ArticleCountService;
import com.zhanghan.zhredistodb.util.wrapper.WrapMapper;import cn.hutool.core.util.IdUtil;@Service
public class ArticleCountServiceImpl implements ArticleCountService {private static Logger logger = LoggerFactory.getLogger(ArticleCountServiceImpl.class);@Autowiredprivate RedisTemplate<String, String> strRedisTemplate;@Autowiredprivate XArticleCountMapper xArticleCountMapper;@Value("${zh.article.count.redis.key:zh}")private String zhArticleCountRedisKey;@Value("#{T(java.lang.Integer).parseInt('${zh..article.read.num:3}')}")private Integer articleReadNum;/*** 记录用户访问记录*/@Overridepublic Object postArticleViews(PostArticleViewsRequest postArticleViewsRequest) {ArticleCountDto articleCountDto = new ArticleCountDto();articleCountDto.setBuNo(IdUtil.simpleUUID());articleCountDto.setCustomerId(postArticleViewsRequest.getCustomerId());articleCountDto.setArticleNo(postArticleViewsRequest.getArticleNo());articleCountDto.setReadTime(new Date());String strArticleCountDto = JSON.toJSONString(articleCountDto);strRedisTemplate.opsForList().rightPush(zhArticleCountRedisKey, strArticleCountDto);return WrapMapper.ok();}/*** 批量将缓存中的数据同步到MySQL*/@Overridepublic Object postBatchRedisToDb() {Date now = new Date();while (true) {List<String> strArticleCountList =strRedisTemplate.opsForList().range(zhArticleCountRedisKey, 0, articleReadNum);if (CollectionUtils.isEmpty(strArticleCountList)) {return WrapMapper.ok();}List<ArticleCountDto> articleCountDtoList = new ArrayList<>();strArticleCountList.stream().forEach(x -> {ArticleCountDto articleCountDto = JSON.parseObject(x, ArticleCountDto.class);articleCountDtoList.add(articleCountDto);});//过滤出本次定时任务之前的缓存中数据,防止死循环List<ArticleCountDto> beforeArticleCountDtoList = articleCountDtoList.stream().filter(x -> x.getReadTime().before(now)).collect(Collectors.toList());if (CollectionUtils.isEmpty(beforeArticleCountDtoList)) {return WrapMapper.ok();}xArticleCountMapper.batchAdd(beforeArticleCountDtoList);Integer delSize = beforeArticleCountDtoList.size();strRedisTemplate.opsForList().trim(zhArticleCountRedisKey, delSize, -1L);}}}

测试

  • 模拟用户请求访问后台(多次请求)
  • 查看缓存中访问数据
  • 模拟定时任务将缓存中数据同步到DB中
  • 这时查看缓存中的数据已经没了
  • 查看数据库表结构

总结

  • 项目中定时任务

    • 问演示方便用http代替定时任务调度;实际项目中用XXL-job,参考:定时任务的选型及改造
    • 定时任务项目中用redis锁防止并发(定时任务调度端多次调度等),参考:Redis实现计数器—接口防刷—升级版(Redis+Lua)
  • 后期运营数据可以从阅读记录表中拉数据进行相关分析
  • 访问量大:可以将MySQL中的阅读记录表定时迁移走(MySQL建历史表,MongoDB等)

Spring Boot实战解决高并发数据入库: Redis 缓存+MySQL 批量入库相关推荐

  1. 猿创征文 | 微服务 Spring Boot 整合Redis 实战开发解决高并发数据缓存

    文章目录 一.什么是 缓存? ⛅为什么用缓存? ⚡如何使用缓存 二.实现一个商家缓存 ⌛环境搭建 ♨️核心源码 ✅测试接口 三.采用 微服务 Spring Boot 注解开启缓存 ✂️@CacheEn ...

  2. 高并发简单解决方案————redis队列缓存+mysql 批量入库(ThinkPhP)

    源码地址:https://github.com/Tinywan/PHP_Experience 问题分析 问题一:要求日志最好入库:但是,直接入库mysql确实扛不住,批量入库没有问题,done.[批量 ...

  3. redis队列缓存 + mysql 批量入库 + php离线整合

    需求背景:有个调用统计日志存储和统计需求,要求存储到mysql中:存储数据高峰能达到日均千万,瓶颈在于直接入库并发太高,可能会把mysql干垮. 问题分析 思考:应用网站架构的衍化过程中,应用最新的框 ...

  4. 彻夜怒肝!Spring Boot+Sentinel+Nacos高并发已撸完,快要裂开了!

    很多人说程序员是最容易实现财富自由的职业,也确实,比如字节 28 岁的程序员郭宇不正是从普通开发一步步做起的吗? 回归行业现状,当开发能力可以满足公司业务需求时,拿到超预期的 Offer 并不算难.最 ...

  5. Java 面试高频题:Spring Boot+Sentinel+Nacos高并发已撸完

    2021都说工作不好找,也是对开发人员的要求变高.前段时间自己有整理了一些Java后端开发面试常问的高频考点问题做成一份PDF文档(1000道高频题),同时也整理一些图文解析及笔记,今天在这免费分享给 ...

  6. Java面试高Spring Boot+Sentinel+Nacos高并发已撸完

    2021都说工作不好找,也是对开发人员的要求变高.前段时间自己有整理了一些Java后端开发面试常问的高频考点问题做成一份PDF文档(1000道高频题),同时也整理一些图文解析及笔记,今天在这免费分享给 ...

  7. Java面试高频题:Spring Boot+JVM+Nacos高并发+高可用已撸完​

    2021都说工作不好找,也是对开发人员的要求变高.前段时间自己有整理了一些Java后端开发面试常问的高频考点问题做成一份PDF文档(1000道高频题),同时也整理一些图文解析及笔记,今天在这免费分享给 ...

  8. Java面试高频题:Spring Boot+Sentinel+Nacos高并发已撸完

    2021都说工作不好找,也是对开发人员的要求变高.前段时间自己有整理了一些Java后端开发面试常问的高频考点问题做成一份PDF文档(1000道高频题),同时也整理一些图文解析及笔记,今天在这免费分享给 ...

  9. 限量!“Java成长笔记”Spring Boot/Sentinel/Nacos高并发

    前言 本文是为了帮大家快速回顾了Java中知识点,这套面试手册涵盖了诸多Java技术栈的面试题和答案,相信可以帮助大家在最短的时间内用作面试复习,能达到事半功倍效果. 本来想将文件上传到github上 ...

最新文章

  1. 边缘计算的三种模式:MEC、微云和雾计算
  2. 事务隔离级别IsolationLevel
  3. python 各种推导式玩法
  4. List集合ArrayList,LinkList
  5. 参数依赖查找(ADL,Argument-dependent lookup)
  6. can通道采样频率_CAN采样点设置为多少合适?设置不对会咋样?
  7. 十、request.getSession( )、reqeust.getSession(false)和 request.getSession(true)
  8. ZooKeeper(四) 使用Redis RedissonLock 实现分布式锁
  9. mysql列的数值型,字符型,日期型
  10. 在线闹钟html代码复制,html5时钟实现代码
  11. c语言枪战游戏代码,FPS射击游戏《林海雪原》完整源代码
  12. 纲要-Java网络爬虫系统性学习与实战(1)
  13. 目前在题库管理和试卷生成方面最好用的共享软件(已更新)
  14. 计算机二级考试怎么练题库,计算机二级考试单选题训练题库
  15. 基于Linux、QT、C++的“别踩白块儿”小游戏
  16. 计算机网络知识点(四) 介质访问控制子层
  17. 力扣编程题-解法汇总
  18. 无线路由器服务器名怎么设置,服务器改无线路由器怎么设置
  19. mysql left_mysql的left函数
  20. 手机S60第三版教程集合

热门文章

  1. 技术无价,“悟”有所值——UCan下午茶这一年
  2. 谷歌安卓之父离职内幕;抖音上线小程序;苹果被罚千万 | 极客头条
  3. 深入浅出 Proguard
  4. 有机晶体数据库_面向Journal of Organic Chemistry作者的晶体学信息文件(CIF)工作流程将于10月6日作出调整...
  5. 经验分享:正则表达式生成器java
  6. html 三大标签,网站三大标签的正确书写
  7. 公办低分二本_河南最适合“二本”考生的30所公办大学,录取分低,考生不要错过...
  8. inl和dnl matlab_请问如何用matlab仿真它的性能, 比如 INL, DNL, SFDR,EOB等等。
  9. 谐波合成法matlab,基于Kaimal谱采用谐波合成法生成脉动风场
  10. java中elapseTime设置新时间,Java ApplicationLike.getApplicationStartElapsedTime方法代码示例...