[Spring MVC 8]高并发实战小Demo

news2025/1/10 11:05:28

本项目基于Spring MVC进行关于点赞项目的开发,从传统的点赞到高并发缓存开发最后到消息队列异步开发,可谓是令人大开眼界。
本篇博客全部代码已经放出,本博客重点是后端操作,所以对于前端就十分简单的页面。讲述了关于Redis,Quartz定时器、ActiveMQ消息队列等相关内容,需要好好掌握。
关于Spring MVC基本上内容也都完结了,后期还继续深化Spring 体系,特别是源码系统,都需要好好掌握。

这里就高并发稍微说明一下,我们常见的高并发场景有:淘宝的双11、春运时的抢票、微博大V的热点新闻等。除了这些典型事情,每秒几十万请求的秒杀系统、每天千万级的订单系统、每天亿级日活的信息流系统等,都可以归为高并发。

而传统的项目大多直接操作数据库,这会造成数据库承受不住,所以引入了Redis进行缓存。

目录

  • 项目概述
    • 数据库表与持久化类
    • DAO层和Mapper映射文件
    • service层和DTO类
    • Controller层和前端页面
  • 传统点赞实现
  • 集成Redis 缓存
    • Redis使用
    • 集成Redis缓存
    • 设计Redis 数据结构
    • 集成 Quartz定时器
  • 集成ActiveMQ
    • 集成ActiveMQ
    • ActiveMQ异步消费

项目概述

在社交网站或者App中,点赞场景有很多,比如微信说说点赞、微博点赞等。普通人发的说说点赞数比较少,所以并发数少,而一些名人发的微博,由于粉丝多,可能短时间点赞数高达数百万。面对如此高并发的点赞,如果没有设计好项目那么会导致服务器和数据库压力过大出现异常。很多公司后端架构必会采取很多措施来解决这种高并发场景,比如引入缓存提升读的性能、使用MQ队列进行异步处理。
本章没有复杂的前端页面和业务,主要是部分功能,理解高并发的架构设计。

数据库表与持久化类

表的内容已经在ay_user数据库中
创建user表(用户),mood表(说说),user_mood_praise_rel 表(点赞关联表)
user和mood是一对多的关系,关联表主要记录用户和说说的关联关系,即哪些说说被哪些用户点赞。

create table user (
    id varchar(32) not null ,
    name varchar(20) default  null,
    account varchar(20) default  null,
    primary key (id)
#     key 'user_name_index' (name) using btree ,
#     key 'user_account_index' (account) using btree
) engine = InnoDB default charset = utf8;

create table mood(
    id varchar(32) not null  ,
    content varchar(256) default null,
    user_id varchar(32) default null,
    publish_time datetime default  null,
    praise_num int(11) default null,
    primary key (id)
)engine = InnoDB default charset = utf8;

create table user_mood_praise_rel (
    id bigint(32) not null  AUTO_INCREMENT,
    user_id varchar(32) default null,
    mood_id varchar(32) default  null,
    primary key (id)
)engine = InnoDB default charset = utf8;

insert into user (id, name, account) values ('1','Jacin','ay');
insert into user (id, name, account) values ('2','ali','a1');

insert into mood (id, content, user_id, publish_time, praise_num) values ('1','Good','1','2022-11-12','100');

在ay.model包创建user表的持久化对象,具体代码如下:

public class User implements Serializable {
    private String id;
    private String name;
   
    private String account;

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }

    public String getAccount() {
        return account;
    }

    public void setAccount(String account) {
        this.account = account;
    }
}

同样的道理,建立mood,user_mood_praise_rel类,和数据库字段类似,这里就不再放代码了。

DAO层和Mapper映射文件

数据库表和持久化类创建完成后,继续创建对应的Dao类和Mapper映射文件。
在ay.dao创建对应的Dao类:
这里注意Repository(一般用于接口层):该注解的作用不只是将类识别为Bean,同时它还能将所标注的类中抛出的数据访问异常封装为 Spring 的数据访问异常类型。 Spring本身提供了一个丰富的并且是与具体的数据访问技术无关的数据访问异常结构,用于封装不同的持久层框架抛出的异常,使得异常独立于底层的框架。

@Repository
public interface UserDao {
    User find(String id);
}

在mapper目录下创建user表对应的UserMapper.xml:

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.ay.dao.UserDao">
    <cache/>
    <resultMap id="userMap" type="com.ay.model.User">
        <id property="id" column="id"/>
        <result property="name" column="name"/>
        <result property="account" column="account"/>
    </resultMap>
    <!-- sql代码块 -->
    <sql id="table_column">
        id,
        name,
        account
    </sql>

    <select id="find" resultMap="userMap">
        select
        <include refid="table_column"/>
        from user
        <where>
            id = #{id}
        </where>
    </select>

</mapper>

其中MoodDao.java:

@Repository
public interface MoodDao {
    List<Mood> findAll();
}

以及UserMoodPraiseRelDao.java:

@Repository
public interface UserMoodPraiseRelDao {
    boolean save(@Param("userMoodPraiseRel")UserMoodPraiseRel userMoodPraiseRel);
}

相关的mapper.xml就不再重复写了。

service层和DTO类

DTO是数据传输对象,用于extends Model类。
通过DTO我们实现了表现层与Model之间的解耦,表现层不引用Model,DTO extends Model层。
ay.dto\MoodDTO.java:

public class MoodDTO extends Mood implements Serializable {
    
    private String userName;

    private String userAccount;

    public String getUserName() {
        return userName;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }

    public String getUserAccount() {
        return userAccount;
    }

    public void setUserAccount(String userAccount) {
        this.userAccount = userAccount;
    }
}

ay.dto\UserDTO.java:

package com.ay.dto;

import com.ay.model.User;
public class UserDTO extends User {
}

MoodDTO和UserDTO用于前端展示用的DTO对象,内容比较简单。
接着在service包下面:UserService.java:

public interface UserService {
    UserDTO find(String id);
}

MoodService.java:

public interface MoodService {
    //传统查询
    List<MoodDTO> findAll();
}

接着实现UserServiceImpl.java:


@Service
public class UserServiceImpl implements UserService {
    @Resource
    private UserDao userDao;
    public UserDTO find(String id) {
        User user = userDao.find(id);
        return converModel2DTO(user);
    }
    private UserDTO converModel2DTO(User user) {
        UserDTO userDTO = new UserDTO();
        userDTO.setId(user.getId());
        userDTO.setAccount(user.getAccount());
        userDTO.setName(user.getName());
        return userDTO;
    }
}

MoodServiveImpl.java:

@Service
public class MoodServiveImpl implements MoodService {
    @Resource
    private MoodDao moodDao;
    @Resource
    private UserDao userDao;


    public List<MoodDTO> findAll() {
        List<Mood> moodList = moodDao.findAll();
        return converModel2DTO(moodList);
    }

    private List<MoodDTO> converModel2DTO(List<Mood> moodList) {
        if (CollectionUtils.isEmpty(moodList)) return Collections.EMPTY_LIST;
        List<MoodDTO> moodDTOList = new ArrayList<MoodDTO>();
        for (Mood mood : moodList) {
            MoodDTO moodDTO = new MoodDTO();
            moodDTO.setId(mood.getId());
            moodDTO.setContent(mood.getContent());
            moodDTO.setPraiseNum(mood.getPraiseNum());
            moodDTO.setPublishTime(mood.getPublishTime());
            moodDTO.setUserId(mood.getUserId());
            moodDTOList.add(moodDTO);
            //设置用户信息
            User user = userDao.find(mood.getUserId());
            moodDTO.setUserName(user.getName());
            moodDTO.setUserAccount(user.getAccount());
        }
        return moodDTOList;
    }
}

然后实现UserMoodPraiseRelService.java:

public interface UserMoodPraiseRelService {

    boolean save(UserMoodPraiseRel userMoodPraiseRel);
}

接着实现类:

@Service
public class UserMoodPraiseRelServiceImpl implements UserMoodPraiseRelService {

    @Resource
    private UserMoodPraiseRelDao userMoodPraiseRelDao;

    public boolean save(UserMoodPraiseRel userMoodPraiseRel) {
        return userMoodPraiseRelDao.save(userMoodPraiseRel);
    }
}

Controller层和前端页面

ay.controller\UserController.java:

@RestController
@RequestMapping("/user")
public class UserController {
    @Resource
    private UserService userService;
}

MoodController.java:

@Controller
@RequestMapping("/mood")
public class MoodController {
    @Resource
    private MoodService moodService;

    @RequestMapping("/findAll")
    public String findAll(Model model) {
        List<MoodDTO> moodDTOList = moodService.findAll();
        model.addAttribute("moods",moodDTOList);
        return "mood";
    }
}

在views/mood.jsp:

<%@page language="java" contentType="text/html; charset=UTF-8"
        pageEncoding="UTF-8" isELIgnored="false" %>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
<%@ taglib prefix="fmt" uri="http://java.sun.com/jstl/fmt" %>
<!DOCTYPE HTML>
<html>
<head>
    <title>Getting Started: Serving Web Content</title>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/>
</head>
<body>

<div id="moods">
    <b>说说列表:</b><br>
    <c:forEach items="${moods}" var="mood">
        ------------------------------------
        <br>
        <b>用户:</b><span id="account">${mood.userName}</span><br>
        <b>说说内容:</b><span id="content">${mood.content}</span><br>
        <b>发表时间:</b>
        <span id="publish_time">
                ${mood.publishTime}
        </span><br>
        <b>点赞数:</b><span id="praise_num">${mood.praiseNum}</span><br>
        <div style="margin-left: 350px">
               <a id="praise" href="/mood/${mood.id}/praise?userId=${mood.userId}"></a>
            
        </div>
    </c:forEach>
</div>
</body>
<script></script>
</html>


进行测试后,浏览器输入80/mood/findAll:
在这里插入图片描述

传统点赞实现

设计思路示例图:
在这里插入图片描述
service层主要做两件事:1.保存点赞用户和被点赞的关联关系,关系保存在rel表中 2.更新说说点赞数。
service层处理过程中,请求数据库获取连接,执行相关的数据库操作归还数据库连接,最终返回数据给用户。
如果在高并发情况下,如果半小时点赞数高达20万,那么QPS高达111(QPS=每秒请求数/事务数量),也就是说后端服务每秒创建111个线程来处理点赞请求。数据库的连接数量有限的会导致响应时间长,处理慢,这就是传统实现的弊端。

代码实现:
MoodMapper.xml:

<select id="findById" resultMap="moodMap">
        select
        <include refid="table_column"/>
        from mood
        <where>
            id = #{id}
        </where>
    </select>

    <update id="update">
        update mood
        <set>
            <if test="mood.content != null and mood.content != ''">
                content = #{mood.content},
            </if>
            <if test="mood.praiseNum != null and mood.praiseNum != ''">
                praise_num = #{mood.praiseNum},
            </if>
        </set>
        WHERE id = #{mood.id}
    </update>

findById用来根据id查询说说实体 mood,update用来更新说说数据。
在MoodDao.java添加:

@Repository
public interface MoodDao {
    List<Mood> findAll();
    boolean update(@Param("mood") Mood mood);

    Mood findById(String id);
}

在MoodService.java添加:

public interface MoodService {
    //传统查询
    List<MoodDTO> findAll();

    //传统点赞
    boolean praiseMood(String userId, String moodId);

    boolean update(@Param("mood") Mood mood);

    Mood findById(String id);

}

MoodService 接口主要添加三个方法,praiseMood()处理用户点赞,update()更新说说内容,findById()查说说:
向MoodServiceImpl.java添加:

@Service
public class MoodServiveImpl implements MoodService {
    @Resource
    private MoodDao moodDao;
    @Resource
    private UserDao userDao;

    @Resource
    private UserMoodPraiseRelDao userMoodPraiseRelDao;
   // DTO处理代码已经省略
  
    public boolean praiseMood(String userId, String moodId) {
        //保存关联关系
        UserMoodPraiseRel userMoodPraiseRel = new UserMoodPraiseRel();
        userMoodPraiseRel.setUserId(userId);
        userMoodPraiseRel.setMoodId(moodId);
        userMoodPraiseRelDao.save(userMoodPraiseRel);
        //更新说说的点赞数量
        Mood mood = this.findById(moodId);
        mood.setPraiseNum(mood.getPraiseNum() + 1);
        this.update(mood);

        return Boolean.TRUE;
    }

    public boolean update(Mood mood) {
        return moodDao.update(mood);
    }

    public Mood findById(String id) {
        return moodDao.findById(id);
    }

}

MoodController.java添加如下代码:

@Controller
@RequestMapping("/mood")
public class MoodController {
    @Resource
    private MoodService moodService;
    // 省略find代码
    @GetMapping(value = "/{moodId}/praise")
    public String praise(Model model, @PathVariable(value = "moodId") String moodId,
                         @RequestParam(value = "userId") String userId) {
        boolean isPraise = moodService.praiseMood(userId, moodId);
        List<MoodDTO> moodDTOList = moodService.findAll();
        model.addAttribute("moods", moodDTOList);
        model.addAttribute("isPraise", isPraise);
        return "mood";
    }
}

mood.jsp代码不变。
下面进行测试:
输入80/mood/findAll:
点击 赞 会将点赞数+1
在这里插入图片描述
注意此时url已经改变。

集成Redis 缓存

传统的点赞功能出现的问题很多:1. 高并发请求下,服务器频繁创建线程 2. 高并发请求下,数据库连接池连接数有限 3.高并发请求下,点赞功能是同步处理
我们引入Redis缓存,每次点赞请求不是直接和MySQL数据库进行交互,而是和Redis进行交互,即把点赞相关的数据保存到Redis缓存,最后通过Quartz 创建定时计划,再把缓存的数据保存到数据库中。
在这里插入图片描述

Redis使用

启动redis服务器:redis-server.exe.打开redis-cli.exe 进入Redis客户端:
端口是6379,固定为Redis端口。
字符串类型的增删改查:

在这里插入图片描述

List 集合的增删改查:
在这里插入图片描述
Set集合的增删改查:
在这里插入图片描述
Hash 集合的增删改查:
在这里插入图片描述
可以在java写个程序测试redis连接性:
使用Java操作Redis需要jedis-2.1.0.jar,下载地址:http://files.cnblogs.com/liuling/jedis-2.1.0.jar.zip

import redis.clients.jedis.Jedis;

public class TestRedis {
    public static void main(String[] args) {
        Jedis jedis = new Jedis("127.0.0.1");
        jedis.set("namc","a1");
    }
}

集成Redis缓存

首先在pom.xml引入所需要的依赖:

<!-- 集成redis -->
        <dependency>
            <groupId>org.springframework.data</groupId>
            <artifactId>spring-data-redis</artifactId>
            <version>${spring.redis.version}</version>
        </dependency>

        <dependency>
            <groupId>org.apache.commons</groupId>
            <artifactId>commons-pool2</artifactId>
            <version>${commons.version}</version>
        </dependency>

        <dependency>
            <groupId>redis.clients</groupId>
            <artifactId>jedis</artifactId>
            <version>${jedis.version}</version>
        </dependency>

接着在resources目录创建redis.properties:
redis密码默认为空

redis.maxIdle=300
redis.minIdle=100
redis.maxWaitMillis=3000
redis.testOnBorrow=true
redis.maxTotal=500
redis.host=127.0.0.1
redis.port=6379
redis.password=

resources创建spring-redis.xml配置文件:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:p="http://www.springframework.org/schema/p"
       xmlns:tx="http://www.springframework.org/schema/tx"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="
      http://www.springframework.org/schema/beans
      http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
      http://www.springframework.org/schema/tx
      http://www.springframework.org/schema/tx/spring-tx-3.0.xsd
      http://www.springframework.org/schema/context
      http://www.springframework.org/schema/context/spring-context-3.0.xsd">

    <context:property-placeholder location="classpath:*.properties"/>
    <!--设置数据池-->
    <bean id="poolConfig" class="redis.clients.jedis.JedisPoolConfig">
        <property name="maxIdle" value="${redis.maxIdle}"></property>
        <property name="minIdle" value="${redis.minIdle}"></property>
        <property name="maxTotal" value="${redis.maxTotal}"></property>
        <property name="maxWaitMillis" value="${redis.maxWaitMillis}"></property>
        <property name="testOnBorrow" value="${redis.testOnBorrow}"></property>
    </bean>
    <!--链接redis-->
    <bean id="redisConnectionFactory" class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory">
        <property name="hostName" value="${redis.host}"></property>
        <property name="port" value="${redis.port}"></property>
        <property name="password" value="${redis.password}"></property>
        <property name="poolConfig" ref="poolConfig"></property>
    </bean>

    <bean id="redisTemplate" class="org.springframework.data.redis.core.RedisTemplate"
          p:connection-factory-ref="redisConnectionFactory">
    </bean>
</beans>

在applicationContext.xml导入spring-redis.xml:

<import resource="spring-redis.xml"/>

配置完成后,在test文件夹进行RedisTest.java进行测试:


public class RedisTest extends  BaseJunit4Test{
    @Resource
    private RedisTemplate redisTemplate;

    @Test
    public void testRedis() {
        redisTemplate.opsForValue().set("name","ay1");
        String name =(String)redisTemplate.opsForValue().get("name");
        System.out.println(name);
    }
}

设计Redis 数据结构

Redis数据结构采用多个Set类型的结合来存放。用Set集合来存放所有被点赞说说的id,key可以是自己约定的唯一的key即可,而value为所有被点赞的说说id,用n个set集合来存放每条说说用户点赞的记录。如果需要获取某个说说被点赞的次数,只要统计set集合size()即可。
在这里插入图片描述
在MoodService.java添加:

	boolean praiseMoodForRedis(String userId, String moodId);

    List<MoodDTO> findAllForRedis();

在MoodServiceImpl.java:

	@Resource
    private RedisTemplate redisTemplate;
	// key命名规范: 项目名称+模块名称+具体内容
    private static final String PRAISE_HASH_KEY = "springmv.mybatis.boot.mood.id.list.key";
     public boolean praiseMoodForRedis(String userId, String moodId) {
        //1.存放到hashset中
        redisTemplate.opsForSet().add(PRAISE_HASH_KEY , moodId);
        //2.存放到set中
        redisTemplate.opsForSet().add(moodId,userId);
        return false;
    }
	@Resource
    private UserService userService;

    public List<MoodDTO> findAllForRedis() {
        List<Mood> moodList = moodDao.findAll();
        if (CollectionUtils.isEmpty(moodList)) {
            return Collections.EMPTY_LIST;
        }
        List<MoodDTO> moodDTOList = new ArrayList<MoodDTO>();
        for (Mood mood : moodList) {
            MoodDTO moodDTO = new MoodDTO();
            moodDTO.setId(mood.getId());
            moodDTO.setUserId(mood.getUserId());
            //right = 总点赞数量 : 数据库的点赞数量 + redis的点赞数量
            moodDTO.setPraiseNum(mood.getPraiseNum() + redisTemplate.opsForSet().size(mood.getId()).intValue());
            moodDTO.setPublishTime(mood.getPublishTime());
            moodDTO.setContent(mood.getContent());
            //通过userID查询用户
            User user = userService.find(mood.getUserId());
            //用户名
            moodDTO.setUserName(user.getName());
            //账户
            moodDTO.setUserAccount(user.getAccount());
            moodDTOList.add(moodDTO);
        }
        return moodDTOList;
    }

MoodServiceImpl 类实现了MoodService 接口中的方法,处理逻辑较为简单:1.保存mood_id到Set集合 2.保存mood_id 和user_id 到Set集合中。
注意这里有多少说说被点赞在Redis缓存就有多少个Set集合。
在MoodController.java:
这里为了简单起见,user_id就直接随机生成了。

@GetMapping(value = "/{moodId}/praiseForRedis")
    public String praiseForRedis(Model model, @PathVariable(value = "moodId") String moodId,
                                 @RequestParam(value = "userId") String userId) {
        //方便使用,随机生成用户id
        Random random = new Random();
        userId = random.nextInt(100) + "";

        boolean isPraise = moodService.praiseMoodForRedis(userId, moodId);
        //查询所有的说说数据
        List<MoodDTO> moodDTOList = moodService.findAllForRedis();
        model.addAttribute("moods", moodDTOList);
        model.addAttribute("isPraise", isPraise);
        return "mood";
    }

将mood.jsp的点赞路径改一下:

<a id="praise" href="/mood/${mood.id}/praiseForRedis?userId=${mood.userId}"></a>

现在运行80/mood/findAll:
左边是相应的mysql:发现确实是不一样的数据(说明此时做到了缓存)
在这里插入图片描述
当你再次输入80/mood/findAll的时候,你会发现点赞数还是143,但是你点击点赞的时候就会变成156,说明数据在redis做到了缓存:
在这里插入图片描述
下面要做的是定时从redis读数据进入数据库保存
可以用Test进行测试:

 @Test
    public void testRedis() {
        //redisTemplate.opsForValue().set("htf", "ayccc");
        Set<String> set = redisTemplate.opsForSet().members("springmv.mybatis.boot.mood.id.list.key");
        System.out.println("value of name is:" + set);
    }

集成 Quartz定时器

Quartz是一个Java 编写的开源任务调度的框架,通过触发器设置作业定时运行规则,控制作业的运行时间。定时器作用很多,比如,定时发送信息和定时生成报表等。
Quartz 框架主要核心组件包括调度器、触发器和作业。调度器作为作业的总指挥,触发器作为作业的操作者,作业为应用的功能模块。
在这里插入图片描述
Job 是一个接口,只有一个execute,被调度的作业需要实现该接口中execute()。
下面在ay\job创建PraiseDataSaveDBJob.java:


@Component

@Configurable
// 相当于配置文件,被Spring 扫描初始化
@EnableScheduling
// 开启对计划任务的支持,在执行的任务上注解@Scheduled,声明是一个计划任务。
public class PraiseDataSaveDBJob {

    //每5秒执行一次
    @Scheduled(cron = "*/60 * *  * * * ")
    public void savePraiseDataToDB() {
        System.out.println("run .....");
    }

    @Resource
    private RedisTemplate redisTemplate;
    private static final String PRAISE_HASH_KEY = "springmv.mybatis.boot.mood.id.list.key";
    @Resource
    private UserMoodPraiseRelService userMoodPraiseRelService;
    @Resource
    private MoodService moodService;

    //每10秒执行一次,真实项目当中,我们可以把定时器的执行计划时间设置长一点
    //比如说每天晚上凌晨2点跑一次。
    @Scheduled(cron = "*/10 * *  * * * ")
    public void savePraiseDataToDB2() {

        //获取所有被点赞的说说id
        Set<String> moods = redisTemplate.opsForSet().members(PRAISE_HASH_KEY);
        if (CollectionUtils.isEmpty(moods)) {
            return;
        }
        for (String moodId : moods) {
            if (redisTemplate.opsForSet().members(moodId) == null) {
                continue;
            } else {
                //通过说说id获取所有点赞的用户id列表
                Set<String> userIds = redisTemplate.opsForSet().members(moodId);
                if (CollectionUtils.isEmpty(userIds)) {
                    continue;
                } else {
                    for (String userId : userIds) {
                        UserMoodPraiseRel userMoodPraiseRel = new UserMoodPraiseRel();
                        userMoodPraiseRel.setMoodId(moodId);
                        userMoodPraiseRel.setUserId(userId);
                        //保存说说与用户关联关系
                        userMoodPraiseRelService.save(userMoodPraiseRel);
                    }
                    Mood mood = moodService.findById(moodId);
                    //更新说说点赞数量
                    //说说的总点赞数量 = redis 点赞数量 + 数据库的点赞数量
                    mood.setPraiseNum(mood.getPraiseNum() + redisTemplate.opsForSet().size(moodId).intValue());
                    moodService.update(mood);
                    //清除缓存数据
                    redisTemplate.delete(moodId);
                }
            }
        }
        //清除缓存数据
        redisTemplate.delete(PRAISE_HASH_KEY);

    }
}

下面开始测试:输入80/mood/findAll,点击点赞按钮会发现10秒后数据库内容将进行更新,注意我开启定时的时候会将缓存数据删除。

集成ActiveMQ

本次解决的是对点赞功能同步处理。消息队列的特点:异步、削峰、解耦。解决方案:
把数据放到消息队列叫做生产者
从消息队列里边取数据叫做消费者
在这里插入图片描述
ActiveMQ安装
MQ:MessageQueue,消息队列,是一个消息的接收和转发的容器,用于消息推送。ActiveMQ是一个开源的消息系统,完全采用Java实现,所以很好支持JMS规范。
直接去官网下载:ActiveMQ
进入win64包打开activemq.bat:
在这里插入图片描述
浏览器输入http://localhost:8161/admin/ 账号密码均为admin
在这里插入图片描述

集成ActiveMQ

首先在pom.xml引入依赖:

<!-- active mq start -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-jms</artifactId>
            <version>${spring.version}</version>
        </dependency>

        <dependency>
            <groupId>org.apache.activemq</groupId>
            <artifactId>activemq-all</artifactId>
            <version>5.11.2</version>
            <exclusions>
                <exclusion>
                    <artifactId>spring-context</artifactId>
                    <groupId>org.springframework</groupId>
                </exclusion>
                <exclusion>
                    <groupId>org.apache.geronimo.specs</groupId>
                    <artifactId>geronimo-jms_1.1_spec</artifactId>
                </exclusion>
            </exclusions>
        </dependency>

        <dependency>
            <groupId>javax.jms</groupId>
            <artifactId>javax.jms-api</artifactId>
            <version>2.0.1</version>
        </dependency>
        <!-- active mq end -->

在resources 目录下创建Active MQ配置文件spring-jms.xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns="http://www.springframework.org/schema/beans" xmlns:context="http://www.springframework.org/schema/context"
       xmlns:jms="http://www.springframework.org/schema/jms"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
                        http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
                        http://www.springframework.org/schema/context
                        http://www.springframework.org/schema/context/spring-context-4.0.xsd
				        http://www.springframework.org/schema/jms
				        http://www.springframework.org/schema/jms/spring-jms-4.0.xsd">
    <bean id="connectionFactory"
          class="org.springframework.jms.connection.CachingConnectionFactory">
        <description>JMS连接工厂</description>
        <property name="targetConnectionFactory">
            <bean class="org.apache.activemq.spring.ActiveMQConnectionFactory">
                <property name="brokerURL" value="${activemq_url}"/>
                <property name="userName" value="${activemq_username}"/>
                <property name="password" value="${activemq_password}"/>
            </bean>
        </property>
        <property name="sessionCacheSize" value="100"/>
    </bean>

    <!-- Spring JmsTemplate 的消息生产者 start-->
    <bean id="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
        <description>队列模式模型</description>
        <constructor-arg ref="connectionFactory"/>
        <property name="receiveTimeout" value="10000"/>

        <property name="pubSubDomain" value="false"/>
    </bean>

    <!-- 消息消费者 start-->
    <!-- 定义Queue监听器 -->
    <jms:listener-container destination-type="queue"
                            container-type="default" connection-factory="connectionFactory"
                            acknowledge="auto">
        <!-- 可写多个监听器 -->
        <jms:listener destination="ay.queue.high.concurrency.praise"
                      ref="moodConsumer"/>
    </jms:listener-container>
    <!-- 消息消费者 end -->
</beans>

resources创建activemq.properties:

### active mq服务器地址
activemq_url=tcp://localhost:61616
### 服务器用户名
activemq_username=admin
### 服务器密码
activemq_password=admin

在applicationContext.xml 引入jms.xml:

    <import resource="spring-jms.xml"/>

ActiveMQ异步消费

在上面集成了ActiveMQ消息中间件,同时开发了相关的配置文件,这一节主要利用ActiveMQ实现点赞功能的异步处理。
在ay\mq目录下创建MoodProducer


@Component
public class MoodProducer {

    @Resource
    private JmsTemplate jmsTemplate;

    private Logger log = Logger.getLogger(this.getClass());

    public void sendMessage(Destination destination, final MoodDTO mood) {
        log.info("生产者--->>>用户id:" + mood.getUserId() + " 给说说id:" + mood.getId() + " 点赞");
        //mood实体需要实现Serializable序列化接口
        jmsTemplate.convertAndSend(destination, mood);
    }
}

提供sendMessage 用来发送消息,方法的第一个参数是destination,主要用来指定队列的名称,第二个参数就是mood说说实体。需要注意的是,MoodDTO需要实现序列化接口。
在ay\mq创建MoodConsumer.java:


public class MoodConsumer implements MessageListener {

    private static final String PRAISE_HASH_KEY = "springmv.mybatis.boot.mood.id.list.key";

    @Resource
    private RedisTemplate redisTemplate;

    private Logger log = Logger.getLogger(this.getClass());

    public void onMessage(Message message) {
        try {
            MoodDTO mood = (MoodDTO) ((ActiveMQObjectMessage) message).getObject();
            //1.存放到set中
            redisTemplate.opsForSet().add(PRAISE_HASH_KEY, mood.getId());
            //2.存放到set中
            redisTemplate.opsForSet().add(mood.getId(), mood.getUserId());
            log.info("消费者--->>>用户id:" + mood.getUserId() + " 给说说id:" + mood.getId() + " 点赞");
        } catch (Exception e) {
            System.out.println(e);
        }
    }
}

实现了MessageListener 接口,完成对消息的监听和接收,消息两种接收方法:同步接收和异步接收。
修改MoodServiceImpl 的praiseMoodForRedis方法:

public boolean praiseMoodForRedis(String userId, String moodId) {
        MoodDTO moodDTO = new MoodDTO();
        moodDTO.setUserId(userId);
        moodDTO.setId(moodId);
        moodProducer.sendMessage(destination, moodDTO);

//        //1.存放到hashset中
//        redisTemplate.opsForSet().add(PRAISE_HASH_KEY , moodId);
//        //2.存放到set中
//        redisTemplate.opsForSet().add(moodId,userId);
        return false;
    }

下面进行测试:输入80/mood/findAll:
表明异步消费开发成功。
在这里插入图片描述

以上就是全部内容了。

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

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

相关文章

软件安装教程1——Neo4j下载与安装

Neo4j的下载地址Neo4j Download Center - Neo4j Graph Data Platform 我下载的是Neo4j社区版&#xff08;免费&#xff09;【企业版收费】 解压后的目录如下&#xff1a; 接下来配置环境变量 进入bin目录&#xff0c;复制路径&#xff1a;E:\neo4j\neo4j-community-5.1.0-win…

决策树——预剪枝和后剪枝

一、 为什么要剪枝 1、未剪枝存在的问题 决策树生成算法递归地产生决策树&#xff0c;直到不能继续下去为止。这样产生的树往往对训练数据的分类很准确&#xff0c;但对未知的测试数据的分类却没有那么准确&#xff0c;即容易出现过拟合现象。解决这个问题的办法是考虑决策树…

【Lua基础 第2章】lua遍历table的方式、运算符、math库、字符串操作方法

文章目录&#x1f4a8;更多相关知识&#x1f447;一、lua遍历table的几种方式&#x1f342;pairs遍历&#x1f342;ipairs遍历&#x1f342;i1,#xxx遍历&#x1f31f;代码演示&#x1f342;pairs 和 ipairs区别二、如何打印出脚本自身的名称三、Lua运算符&#x1f538;算术运算…

微服务治理-含服务线上稳定性保障建设治理

微服务的概念 任何组织在设计一套系统&#xff08;广义概念上的系统&#xff09;时&#xff0c;所交付的设计方案在结构上都与该组织的沟通结构保持一致。 —— 康威定律 微服务是一种研发模式。换句话理解上面这句康威定律&#xff0c;就是说 一旦企业决定采用微服务架构&am…

Js逆向教程-12FuckJs

Js逆向教程-12FuckJs 它利用了js的语法特性&#xff1a; 一、特性1 任何一个js类型的变量结果 加上一个字符串 &#xff0c;只会变成字符串。 数组加上字符串&#xff1a; [0]"" 0true加上字符串 true "" true数字加上字符串 1"" 1二、特性…

14天学习训练营之 初识Pygame

目录 学习知识点 PyGame 之第一个 PyGame 程序 导入模块 初始化 ​​1.screen 2. 游戏业务 学习笔记 当 init () 的时候&#xff0c;它在干什么&#xff1f; init () 实际上检查了哪些东西呢&#xff1f; 它到底 init 了哪些子模块&#xff1f; 总结 14天学习训练营导…

2023年计算机毕设选题推荐

同学们好&#xff0c;这里是海浪学长的毕设系列文章&#xff01; 对毕设有任何疑问都可以问学长哦! 大四是整个大学期间最忙碌的时光,一边要忙着准备考研,考公,考教资或者实习为毕业后面临的就业升学做准备,一边要为毕业设计耗费大量精力。近几年各个学校要求的毕设项目越来越…

·工业 4.0 和第四次工业革命详细介绍

工业 4.0 是制造/生产及相关行业和价值创造过程的数字化转型。 目录 工业 4.0 指南 工业 4.0 与第四次工业革命互换使用&#xff0c;代表了工业价值链组织和控制的新阶段。 网络实体系统构成了工业 4.0 的基础&#xff08;例如&#xff0c;「智慧机器」&#xff09;。他们使用…

基于SpringBoot+Vue的疫苗接种管理系统

末尾获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端&#xff1a;SpringBoot 前端&#xff1a;Vue 数据库&#xff1a;MySQL5.7 数据库管理工具&#xff1a;Navicat 12 服务器&#xff1a;Tomcat8.5 开发软件&#xff1a;IDEA / MyEclipse 是否Maven项…

实验二 帧中继协议配置

计算机网络实验实验二 帧中继协议配置一、实验目的二、实验内容三、实验条件四、实验步骤4.1 连接帧中继交换网4.2 创建DLCI4.3 创建串行接口间的虚电路映射关系4.4 配置路由器的串行接口七、思考题实验二 帧中继协议配置 一、实验目的 掌握路由器上配置帧中继协议的方法 掌握…

SSM整合(一)

SSM整合之简单使用通用mapper 1.准备工作 1.1 在java文件夹下面创建所需要的目录 1.2 导入SSM整合时所需要的所有依赖 <properties><!--这个是统一一些spring插件的包名,避免因为版本不一样而报错--><spring.version>5.3.22</spring.version></p…

SAP S4 FI 后台详细配置教程文档 PART2 (财务会计的基本设置篇)

本篇是系列文章的第二部分&#xff0c;目标是家在配置“字段状态变式”和“年度与期间的配置” 目录 1、 字段状态变式 1.1定义字段状态变式 1.2 向字段状态变式分配公司代码 2、会计年度与记账期间 2.1维护会计年度变式 2.2 向一个会计年度变式分配公司代码 2.3定义未结…

服务器虚拟化有什么好处

服务器虚拟化是一种逻辑角度出发的资源配置技术&#xff0c;是物理实际的逻辑抽象。对于用户&#xff0c;虚拟化技术实现了软件跟硬件分离&#xff0c;用户不需要考虑后台的具体硬件实现&#xff0c;而只需在虚拟层环境上运行自己的系统和软件。 说起服务器虚拟化这个技术&…

你的新进程是如何被内核调度执行到的?(下)

接上文你的新进程是如何被内核调度执行到的&#xff1f;&#xff08;上&#xff09; 四、新进程加入调度 进程在 copy_process 创建完毕后&#xff0c;通过调用 wake_up_new_task 将新进程加入到就绪队列中&#xff0c;等待调度器调度。 //file:kernel/fork.c long do_fork(.…

表白墙服务器版【交互接口、服务器端代码、前端代码、数据存入文件/数据库】

文章目录 一、准备工作二、约定前后端交互接口三、实现服务器端代码 四、调整前端页面代码五、数据存入文件六、数据存入数据库一、准备工作 1) 创建 maven 项目2) 创建必要的目录 webapp, WEB-INF, web.xml&#xff1b;web.xml如下&#xff1a;<!DOCTYPE web-app PUBLIC&qu…

家居行业如何实现智能化?快解析来助力

什么是智能家居&#xff1f;主要是指利用先进的电子通信技术&#xff0c;将居家生活有关的各个子系统有机结合在一起&#xff0c;通过网络化便可以对这些系统进行智能控制与管理。智能家居概念之所以逐渐普及&#xff0c;得益于物联网、大数据、人工智能等新兴技术的进步。智能…

科学计算模型 Numpy 详解

本文主要介绍Numpy&#xff0c;并试图对其进行一个详尽的介绍。 通过阅读本文&#xff0c;你可以&#xff1a; 了解什么是 Numpy掌握如何使 Numpy 操作数组&#xff0c;如创建数组、改变数组的维度、拼接和分隔数组等掌握 Numpy 的常用函数&#xff0c;如数组存取函数、加权平均…

表关联查询

表关联查询 1.表别名 当表的名字很长或者执行一些特殊查询时&#xff0c;为了方便操作或者需要多次使用相同的表时&#xff0c;可以为表指定别名&#xff0c;以替代表原来的名称。 在为表取别名时&#xff0c;要保证不能与数据库中的其他表的名称冲突。 对单表做简单的别名查询…

能否通过手机号查询他人位置及技术实现(省流:不能)

前言 &#x1f340;作者简介&#xff1a;被吉师散养、喜欢前端、学过后端、练过CTF、玩过DOS、不喜欢java的不知名学生。 &#x1f341;个人主页&#xff1a;红中 &#x1fad2;每日emo&#xff1a;纪念我死去的爱情 &#x1f342;灵感来源&#xff1a;艺术源于生活&#xff0c…

SpringBoot SpringBoot 开发实用篇 5 整合第三方技术 5.2 Spring 缓存使用方式

SpringBoot 【黑马程序员SpringBoot2全套视频教程&#xff0c;springboot零基础到项目实战&#xff08;spring boot2完整版&#xff09;】 SpringBoot 开发实用篇 文章目录SpringBootSpringBoot 开发实用篇5 整合第三方技术5.2 Spring 缓存使用方式5.2.1 Spring 缓存使用5.2.…