【wiki知识库】09.欢迎页面展示(浏览量统计)SpringBoot部分

news2024/11/14 7:38:04

🍊 编程有易不绕弯,成长之路不孤单!

大家好,我是熊哈哈,这个项目从我接手到现在有了两个多月的时间了吧,其实本来我在七月初就做完的吧,但是六月份的时候生病了,在家里休息了一个月的时间,后来回到学校考试,考完试之后就出来实习了,这个项目也就一直拖了下去,最近时间够了就在努力的去完成这个项目。

总体来说这个项目不是很难,有很多的东西不够细节,处理的也不是很到位,请大家谅解一下。在我考虑到的不充分的地方一个就是在删除电子书的时候,并没有删除电子书对应存在数据库当中的文章,还有就是在登录的时候,密码加密应该在前端就开始加密了,不应该传到后端在进行加密处理,其实还有很多不到位的地方需要大家去查找,我在提示一下,有没有考虑到为接口添加事物?

此外,项目还可以有改进部分,一个是添加AOP做日志管理,还有自定义异常类,这些都是很重要的模块,在实际的开发当中,一定不要像代码当中的那么随意。其他的我还没有想到,不过我记得我之前说过这个项目的权限校验不太合理,有兴趣的可以了解一下RBAC还有SpringSecurity。

目录

🍊 编程有易不绕弯,成长之路不孤单!

一、今日目标

二、SpringBoot代码修改

2.1 新增IpUtil

2.2 添加访问量统计功能

2.3 新增点赞功能

2.4 实现数据统计

2.4.1 定时更新ebook数据

2.4.2 定时更新ebook_snapshot

2.5 展示总浏览量和30天内的数据变化


一、今日目标

上篇文章链接:【wiki知识库】08.添加用户登录功能--后端SpringBoot部分-CSDN博客

上篇文章做了登录功能的后端逻辑,实现了登录拦截还有权限校验部分,为网站提供了一定的安全性保障,在这篇文章就要实现最后的部分:浏览量的统计和点赞功能。

这一部分就是纯Sql,还有自动化任务。

二、SpringBoot代码修改

2.1 新增IpUtil

这个工具类的作用就是在你访问接口的时候,可以获取到你的真实IP。

public class IpUtil {
    private static final String UNKNOWN = "unknown";
    private static final String LOCALHOST = "127.0.0.1";
    private static final String SEPARATOR = ",";

    public static String getIpAddr(HttpServletRequest request) {
        System.out.println(request);
        String ipAddress;
        try {
            ipAddress = request.getHeader("x-forwarded-for");
            if (ipAddress == null || ipAddress.length() == 0 || UNKNOWN.equalsIgnoreCase(ipAddress)) {
                ipAddress = request.getHeader("Proxy-Client-IP");
            }
            if (ipAddress == null || ipAddress.length() == 0 || UNKNOWN.equalsIgnoreCase(ipAddress)) {
                ipAddress = request.getHeader("WL-Proxy-Client-IP");
            }
            if (ipAddress == null || ipAddress.length() == 0 || UNKNOWN.equalsIgnoreCase(ipAddress)) {
                ipAddress = request.getRemoteAddr();
                if (LOCALHOST.equals(ipAddress)) {
                    InetAddress inet = null;
                    try {
                        inet = InetAddress.getLocalHost();
                    } catch (UnknownHostException e) {
                        e.printStackTrace();
                    }
                    ipAddress = inet.getHostAddress();
                }
            }
            // 对于通过多个代理的情况,第一个IP为客户端真实IP,多个IP按照','分割
            // "***.***.***.***".length()
            if (ipAddress != null && ipAddress.length() > 15) {
                if (ipAddress.indexOf(SEPARATOR) > 0) {
                    ipAddress = ipAddress.substring(0, ipAddress.indexOf(","));
                }
            }
        } catch (Exception e) {
            ipAddress = "";
        }
        return ipAddress;
    }
}

2.2 添加访问量统计功能

这个功能一说出来你知道要在哪里添加吗?就是我们之前写过的fint-content接口,当我们点开一篇文章查看的时候,对应的我们要为这一篇文章增加访问量。

找打这个接口,然后修改代码。

    /**
     * 查找某个doc的content内容
     * @param id content的id
     * @return
     */
    @GetMapping("/find-content/{id}")
    public CommonResp findContent(@PathVariable Long id) {
        Content content = contentService.getById(id);
        // 查找的同时更新阅读数
        docService.increaseViewCount(id);
        String message = content.getContent();
        return new CommonResp(true,"查找成功",message);
    }

这是在DocServiceImpl中添加的,代码很简单,压力给到mapper。

public void increaseViewCount(Long id) {
        docMapper.increaseViewCount(id);
    }

走带mapper中,就是要执行我们自己写的sql语句,在这之前大家先来考虑一个问题,为了增加浏览量我这里给大家两种方案。

  1. 通过文档的id查询出文档的信息,然后修改文档的浏览量,然后在存回去。
  2. 通过文档的id直接找到对应的数据,然后直接修改浏览量。

不用想也是第二种。但是一定要注意,如果要执行自定义的Sql,一定要把xml文件放到resources目录下的mapper中。

<update id="increaseViewCount">
        update doc set view_count = view_count + 1 where id = #{id}
</update>

2.3 新增点赞功能

这个功能还是在DocController中添加的。

对应的接口代码如下。

/**
     * 给文章点赞
     * @param id 传入的doc的id
     * @return
     */
    @GetMapping("/vote/{id}")
    public CommonResp vote(@PathVariable Long id) {
        docService.increaseVoteCount(id);
        return  new CommonResp(true,"点赞成功",null);
    }

DocServiceImpl中新增代码。这段代码中,获取了HttpServletRequest对象,通过这个对象我们就可以拿到用户访问时的IP,为什么需要用户的IP?

你有没有经历过这种情况,当你访问某一篇博客的时候,或者看一个视频的时候,在你点赞了之后你就不能够在点赞了,或者你在点赞就是取消点赞了,我们这里解析用户IP也是一样的道理,我们要把当前点赞用户的IP存放在Redis当中,当用户点过一次之后,我们就设置24小时内不能够在点赞。为什么一定要IP呢?因为网站用户不需要账号登陆。

现在你知道了IpUtil的作用了吗。

 public void increaseVoteCount(Long id) {
        ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
        HttpServletRequest request = attributes.getRequest();
        // 通过IPUtil获取远程访问的IP
        String ip = IpUtil.getIpAddr(request);
        // 判断用户是否已经投票
        if (redisUtil.validateRepeat("DOC_VOTE_" + id + "_" + ip, 60 * 60 * 24)) {
            docMapper.increaseVoteCount(id);
        } else {
            throw new RuntimeException("您今日已经投过票了");
        }
        docMapper.increaseVoteCount(id);
    }

2.4 实现数据统计

想一下,如果这个功能让你来实现,你会从哪里入手?

先来看看我们的数据库的表格吧。我们点开doc、ebook后都能看到有浏览量的统计,但不同的是doc记录的是每一篇文档的浏览量,而ebook记录的是某个电子书的浏览量,至于下边的ebook_snapshot意味着每一日的电子书快照,什么意思呢?就是每一天当中每一本电子书的总浏览量和今日增长的浏览量。

再回想我们的代码,我们只有在代码中添加了有关doc的浏览量统计,那我们改如何同步三个表格呢?

想想看,doc统计的是文档的浏览量,文档是带有ebook的id的,我们只要把某个ebook下的doc做一个统计就好了,这样就可以的到ebook中的浏览量数据了。

那么ebook_snapshot中的数据呢?就像是下边的图所示。

view_count是电子书访问的总浏览量,这个我们在ebook当中就已经有了,至于vote_count和view_count同理。

那么view_increase呢?这个数据记录着今日访问量。

 我给大家标注了一下,现在不难猜了吧,今日的当前总访问量和昨天截止的访问量相减就是今日访问量了。

 好了知道了这些就会让你更容易地理解Sql怎么写。


2.4.1 定时更新ebook数据

这是一个自动化任务,过一段时间ebook中的数据就要和doc中的数据进行同步。

@Component
public class EbookInfoTask {
    private static final Logger LOG = LoggerFactory.getLogger(EbookInfoTask.class);
    @Resource
    private DocService docService;
    @Resource
    private SnowFlake snowFlake;
    /**
     * 每30秒更新电子书信息
     */
    @Scheduled(cron = "5/30 * * * * ?")
    public void cron() {
        // 增加日志流水号
        MDC.put("LOG_ID", String.valueOf(snowFlake.nextId()));
        LOG.info("更新电子书下的文档数据开始");
        long start = System.currentTimeMillis();
        docService.updateEbookInfo();
        LOG.info("更新电子书下的文档数据结束,耗时:{}毫秒", System.currentTimeMillis() - start);
    }
}

service中的代码如下,这个就好理解了。

    <update id="updateEbookInfo">
        update ebook t1, (select ebook_id, count(1) doc_count, sum(view_count) view_count, 
          sum(vote_count) vote_count
        from doc group by ebook_id) t2
        set t1.doc_count = t2.doc_count, t1.view_count = t2.view_count, t1.vote_count = 
          t2.vote_count
        where t1.id = t2.ebook_id
    </update>

2.4.2 定时更新ebook_snapshot

@Component
public class EbookSnapshotTask {

    private static final Logger LOG = LoggerFactory.getLogger(EbookSnapshotTask.class);
    @Resource
    private EbookSnapshotService ebookSnapshotService;
    @Resource
    private SnowFlake snowFlake;
    /**
     * 自定义cron表达式跑批
     * 只有等上一次执行完成,下一次才会在下一个时间点执行,错过就错过
     */
    @Scheduled(cron = "0 0/1 * * * ?")
    public void doSnapshot() {
        // 增加日志流水号
        MDC.put("LOG_ID", String.valueOf(snowFlake.nextId()));
        LOG.info("生成今日电子书快照开始");
        Long start = System.currentTimeMillis();
        ebookSnapshotService.genSnapshot();
        LOG.info("生成今日电子书快照结束,耗时:{}毫秒", System.currentTimeMillis() - start);
    }
}

下边直接给出大家sql语句。

<update id="genSnapshot">
        INSERT INTO ebook_snapshot(ebook_id, `date`, view_count, vote_count, view_increase, vote_increase)
        SELECT t1.id, CURDATE(), 0, 0, 0, 0
        FROM ebook t1
        WHERE NOT EXISTS (
                SELECT 1
                FROM ebook_snapshot t2
                WHERE t1.id = t2.ebook_id
                  AND t2.`date` = CURDATE()
            );

        UPDATE ebook_snapshot t1
            JOIN ebook t2 ON t1.ebook_id = t2.id
        SET t1.view_count = t2.view_count,
            t1.vote_count = t2.vote_count
        WHERE t1.`date` = CURDATE();

        UPDATE ebook_snapshot t1
            LEFT JOIN (
                SELECT ebook_id, view_count, vote_count
                FROM ebook_snapshot
                WHERE `date` = DATE_SUB(CURDATE(), INTERVAL 1 DAY)
            ) t2 ON t1.ebook_id = t2.ebook_id
        SET t1.view_increase = (t1.view_count - IFNULL(t2.view_count, 0)),
            t1.vote_increase = (t1.vote_count - IFNULL(t2.vote_count, 0))
        WHERE t1.`date` = CURDATE();
    </update>

不过你也别被吓到,这个update中一共有三个sql,现在来一条一条分析。

这是一条插入语句目的是什么呢,先看where语句,这是一个带有条件的插入sql,当我们从ebook_snapshot中查找数据时,如果没有发现日期是今天,并且存在于ebook中的数据时候就会执行,说白了就是看一下在当天有没有往这个表格中插入ebook的统计信息,没有的话就插进去。

INSERT INTO ebook_snapshot(ebook_id, `date`, view_count, vote_count, view_increase, vote_increase)
        SELECT t1.id, CURDATE(), 0, 0, 0, 0
        FROM ebook t1
        WHERE NOT EXISTS (
                SELECT 1
                FROM ebook_snapshot t2
                WHERE t1.id = t2.ebook_id
                  AND t2.`date` = CURDATE()
            );

再来看下一条,这是一条更新语句,更新的是表格当中的电子书总浏览量和总点赞量,注意是当天的数据。

 UPDATE ebook_snapshot t1
            JOIN ebook t2 ON t1.ebook_id = t2.id
        SET t1.view_count = t2.view_count,
            t1.vote_count = t2.vote_count
        WHERE t1.`date` = CURDATE();

来看最后一条,还是更新语句,只不过更新的是当天的每个电子书的当日访问量和点赞量。

两张ebook_snapshot表格做连接,拿到一张表格中今天所有电子书的数据,还有昨天所有电子书的数据,然后更新今天所有电子书的view_increase和vote_increase,数值就是前边说的相减。

 UPDATE ebook_snapshot t1
            LEFT JOIN (
                SELECT ebook_id, view_count, vote_count
                FROM ebook_snapshot
                WHERE `date` = DATE_SUB(CURDATE(), INTERVAL 1 DAY)
            ) t2 ON t1.ebook_id = t2.ebook_id
        SET t1.view_increase = (t1.view_count - IFNULL(t2.view_count, 0)),
            t1.vote_increase = (t1.vote_count - IFNULL(t2.vote_count, 0))
        WHERE t1.`date` = CURDATE();

2.5 展示总浏览量和30天内的数据变化

@RestController
@RequestMapping("/ebook-snapshot")
public class EbookSnapshotController {

    @Resource
    private EbookSnapshotService ebookSnapshotService;

    @GetMapping("/get-statistic")
    public CommonResp getStatistic() {
        List<StatisticVo> statisticResp = ebookSnapshotService.getStatistic();
        return new CommonResp<>(true,"success",statisticResp);
    }

    @GetMapping("/get-30-statistic")
    public CommonResp get30Statistic() {
        List<StatisticVo> statisticResp = ebookSnapshotService.get30Statistic();
        return new CommonResp<>(true,"success",statisticResp);
    }
}

话不多说,我们直接上sql,因为代码里直接一路调到mapper了。

这个是统计总浏览量和总点赞量的,我这里统计多了,其实直接让date等于curdate()就可以了。

<select id="getStatistic" resultType="com.my.hawiki.vo.StatisticVo">
        select
            t1.`date` as `date`,
            sum(t1.view_count) as viewCount,
            sum(t1.vote_count) as voteCount,
            sum(t1.view_increase) as viewIncrease,
            sum(t1.vote_increase) as voteIncrease
        from
            ebook_snapshot t1
        where
            t1.`date` >= date_sub(curdate(), interval 1 day)
        group by
            t1.`date`
        order by
            t1.`date` desc;
    </select>

这个呢就是统计三十日内的数据信息,统计的是每一天的浏览量和点赞数,但是不包括当日。 

 <select id="get30Statistic" resultType="com.my.hawiki.vo.StatisticVo">
        select
            t1.`date` as `date`,
            sum(t1.view_increase) as viewIncrease,
            sum(t1.vote_increase) as voteIncrease
        from
            ebook_snapshot t1
        where
            t1.`date` between date_sub(curdate(), interval 30 day) and date_sub(curdate(), interval 1 day)
        group by
            t1.`date`
        order by
            t1.`date` asc;
    </select>

🍊 🍊 🍊 到了这里也就结束了,如果在代码上有问题或者想要获取源码的同学们可以私信联系我,或者联系易编橙。🍊 🍊 🍊

易编橙·终身成长社群,相遇已是上上签!-CSDN博客

文章底部有链接

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2041698.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【和可被 K 整除的子数组】python刷题记录

R4-前缀和专题 class Solution:def subarraysDivByK(self, nums: List[int], k: int) -> int:ret0# 存储当前位置的上一个位置的前缀和的余数加上当前位置的值对K的余数pre_mod0dictdefaultdict(int)dict[0]1for i in range(len(nums)):pre_mod(pre_modnums[i])%k# 如果能在…

Kubernetes 基础概念介绍

1. 应用部署方式 传统部署&#xff1a;互联网早期&#xff0c;会直接将应用程序部署在物理机上 优点&#xff1a;简单&#xff0c;不需要其它技术的参与 缺点&#xff1a;不能为应用程序定义资源使用边界&#xff0c;很难合理地分配计算资源&#xff0c;而且程序之间容易产生影…

投票小程序App功能开发源码技术实现

随着移动互联网的快速发展&#xff0c;小程序作为一种轻量级应用&#xff0c;因其无需安装、即用即走的特点&#xff0c;在各类应用场景中迅速普及。投票小程序作为其中的一种&#xff0c;因其便捷性和实用性&#xff0c;广泛应用于各类活动、问卷调查及意见收集中。本文将围绕…

Linux-Shell入门-05

1.Shell的概念 1.1 什么是shell Shell脚本语言是实现Linux/UNIX系统管理及自W动化运维所必备的重要工具&#xff0c; Linux/UNIX系统的底层及基础应用软件的核心大都涉及Shell脚本的内容。Shell是一种编程语言, 它像其它编程语言如: C, Java, Python等一样也有变量/函数/运算符…

上海“创投十九条”明确政府引导基金带动作用,银行如何挖掘投贷联动增长潜力?

发展创业投资是促进科技、产业、金融良性循环的重要举措。为促进创业投资高质量发展&#xff0c;近几个月来&#xff0c;“国创投十七条”“上海创投十九条”等政策陆续发布&#xff0c;明确指出发挥政府引导基金带动作用&#xff0c;进一步加大对战略性新兴产业和未来产业的支…

空指针异常(NullPointerException)以及解决方案

之前我们提到过&#xff0c;在学习数组这一篇章时&#xff0c;有两种运行时异常&#xff0c;可能会反复的出现,首先第一个就是我们之前所讲述的&#xff0c;数组下标越界异常&#xff08;ArrayIndexOutOfBoundsException&#xff09;&#xff0c;如需查看详情&#xff0c;可跳转…

tekton通过ceph挂载node_modules的时候报错failed to execute command: copying dir: symlink

分析&#xff1a; 如果ceph的mountPath和workingDir路径一致的话&#xff0c;就会报错。 解决&#xff1a;node_modules挂载到/workspace下&#xff0c;workingDir的代码mv到/workspace下进行构建。

Spring Boot和OCR构建车牌识别系统

​ 博客主页: 南来_北往 系列专栏&#xff1a;Spring Boot实战 OCR介绍 OCR&#xff08;Optical Character Recognition&#xff09;是光学字符识别技术的缩写&#xff0c;它能够将图像中的文本转换为机器可读和编辑的数字文本格式。这种技术广泛应用于数据输入、文档管理…

⼆叉树选择题

⼆叉树选择题 本篇文章是初阶二叉树的收尾&#xff0c;旨在进一步加深对于二叉树性质的理解&#xff0c;祝你有一个愉快的学习之旅&#xff01; &#x1f4a1; ⼆叉树性质 1&#xff09;对任何⼀棵⼆叉树, 如果度为 0 其叶结点个数为 n0 , 度为 2 的分⽀结点个数为 n2 ,则有…

Unity | AmplifyShaderEditor插件基础(第二集:模版说明)

目录 一、前言 二、核心模版和URP模版 1.区别介绍 2.自己的模版 三、输出节点 1.界面 2.打开OutPut 3.ShderType 4.ShaderName 5.Shader大块内容 6.修改内容 四、预告 一、前言 内容全部基于以下链接基础以上讲的。 Unity | Shader基础知识&#xff08;什么是shader…

JSON与EXL文件互转

功能&#xff1a;实现json到excel文件的相互转换(支持json多选版) 目的&#xff1a;编码与语言对应&#xff0c;方便大家使用 页面设计&#xff1a; 介绍&#xff1a; 1.选择文件栏目选择想要转换的文件 2.生成路径是转换后文件所在目录 3.小方框勾选与不勾选分别代表exl到…

Apache Doris 中Compaction问题分析和典型案例

说明 此文档主要说明一些常见compaction问题的排查思路和临时处理手段。这些问题包括 Compaction socre高Compaction失败compaction占用资源多Compaction core 如果问题紧急&#xff0c;可联系社区同学处理 如果阅读中有问题&#xff0c;可以反馈给社区同学。 1 compaction …

VulnHub:BlueMoon

准备工作 靶机下载地址&#xff0c;下载完成后使用virtualbox打开虚拟机&#xff0c;网络设置修改为桥接。 信息收集 主机发现 攻击机ip&#xff1a;192.168.31.218&#xff0c;nmap扫描攻击机同网段存活主机。 nmap 192.168.31.0/24 -Pn -T4 目标靶机ip为&#xff1a;192…

Docker最佳实践进阶(一):Dockerfile介绍使用

大家好&#xff0c;上一个系列我们使用docker安装了一系列的基础服务&#xff0c;但在实际开发过程中这样一个个的安装以及繁杂命令不仅仅浪费时间&#xff0c;更是容易遗忘&#xff0c;下面我们进行Docker的进阶教程&#xff0c;帮助我们更快速的部署和演示项目。 一、什么是…

免费抠图项目

零元部署一个产品落地页 最近花了点时间&#xff0c;为我之前写的AI抠图项目&#xff0c;写了个产品落地页&#xff0c;效果如下。 项目使用tailwindcss和vue3实现&#xff0c;支持主题切换、中英文切换等功能。 更多功能可以点击Matting Website查看。 部署这个页面几乎是0元…

Linux文件系统之RAID

文章目录 1、前言2、RAID特性简述3、RAID实现方式4、RAID级别①RAID-0②RAID-1③RAID-5④RAID-6⑤RAID-10和RAID-01 1、前言 RAID全称Redundant Arrays of Inexpensive Disks / Redundant Arrays of Independent Disks&#xff0c;即独立冗余磁盘阵列。RAID可以通过相关技术&a…

纷享销客CRM AI产品架构概览、产品特色

一、纷享销客CRM AI产品架构概览 纷享AI平台架构分为三个主要层次&#xff1a;AI基础设施层、AI平台层和AI应用层。每个层次都由一系列功能模块组成&#xff0c;旨在为客户提供强大的技术支持和灵活的解决方案。 1.Al基础设施层 AI基础设施层是整个AI平台的底层支撑&#xff…

Spark MLlib 特征工程(上)

文章目录 Spark MLlib 特征工程(上)特征工程预处理 Encoding:StringIndexer特征构建:VectorAssembler特征选择:ChiSqSelector归一化:MinMaxScaler模型训练总结Spark MLlib 特征工程(上) 前面我们一起构建了一个简单的线性回归模型,来预测美国爱荷华州的房价。从模型效果来…

【高等代数笔记】002.高等代数研究对象(二)

1. 高等代数的研究对象 1.4 一元高次方程的求根 a n x n a n − 1 x n − 1 . . . a 1 x a 0 0 a_{n}x^{n}a_{n-1}x^{n-1}...a_{1}xa_{0}0 an​xnan−1​xn−1...a1​xa0​0 等式左边是一元多项式。 所有一元多项式组成的集合称为一元多项式环。

在亚马逊云科技上安全、合规地创建AI大模型训练基础设施并开发AI应用服务

项目简介&#xff1a; 小李哥将继续每天介绍一个基于亚马逊云科技AWS云计算平台的全球前沿AI技术解决方案&#xff0c;帮助大家快速了解国际上最热门的云计算平台亚马逊云科技AWS AI最佳实践&#xff0c;并应用到自己的日常工作里。 本次介绍的是如何在亚马逊云科技利用Servi…