在springboot项目中使用rocketmq消息队列实战

news2025/1/12 0:48:14

rocketmq环境搭建

在docket环境下安装部署rocketmq的方法记录在上一篇文章中。

(31条消息) docker环境下搭建rocketmq集群_haohulala的博客-CSDN博客

这种方式不一定是最好的,但是我用这种方式可以成功搭建rocketmq开发环境。 

项目架构

我们需要在springboot中创建两个module,分别是生产者和消费者。

项目的架构非常简单

生产者使用web爬虫从网络中获取信息,然后将信息推送给rocketmq,消费者从rocketmq中获取消息,然后再存储到mysql数据库中。

在这样的案例中,消息队列起到了一个同步生产者和消费者的作用,因为我们的生产者可以有非常多的线程一起进行web爬虫,然后可能会生产非常多的消息,如果没有rocketmq的话,消费者可能会消费不过来,有了rocketmq,生产者和消费者就可以按照自己的节奏放心工作了。

下面我们来创建springboot项目。

首先创建一个空的springboot项目,框中文件都可以删除。

将上面的文件删除后,我们需要建两个module

建立的module还是springboot项目

选择使用maven构建项目,我使用的java版本是jdk1.8

然后可以选择需要用到的start包,这里由于我使用的是jdk1.8,所以对应的springboot版本应该是2开头的

修改pom.xml的内容

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.7.10</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.example</groupId>
    <artifactId>demo</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>rocketmq-client</name>
    <description>Client of rocketmq</description>
    <properties>
        <java.version>1.8</java.version>
    </properties>
    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <dependency>
            <groupId>org.apache.rocketmq</groupId>
            <artifactId>rocketmq-spring-boot-starter</artifactId>
            <version>2.0.4</version>
        </dependency>


        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>

        <!-- 爬虫相关的包 -->
        <dependency>
            <groupId>com.squareup.okhttp3</groupId>
            <artifactId>okhttp</artifactId>
            <version>3.10.0</version>
        </dependency>


        <dependency>
            <groupId>org.jsoup</groupId>
            <artifactId>jsoup</artifactId>
            <version>1.11.3</version>
        </dependency>

        <dependency>
            <!-- fastjson -->
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.47</version>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>


            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-resources-plugin</artifactId>
                <version>2.4.3</version>
            </plugin>

            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
                <version>2.22.2</version>
                <configuration>
                    <skipTests>true</skipTests>
                </configuration>
            </plugin>
        </plugins>
    </build>

</project>

如果在启动项目的时候报错,别忘了在maven中将下面这个选项勾上

可以写一个简单的接口测试一下项目是否启动成功

如果接口调通了,那么就证明项目已经启动成功,接着就是使用同样的方法构建消费者

消费者的pom.xml的内容和生产者稍微有点不同,主要是添加了mysql和mybaits的支持

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.7.10</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.example</groupId>
    <artifactId>demo</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>consumer</name>
    <description>a consumer of rocketmq</description>
    <properties>
        <java.version>1.8</java.version>
    </properties>
    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <dependency>
            <groupId>org.apache.rocketmq</groupId>
            <artifactId>rocketmq-spring-boot-starter</artifactId>
            <version>2.0.4</version>
        </dependency>

        <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>2.1.0</version>
        </dependency>

        <dependency>
            <groupId>com.mysql</groupId>
            <artifactId>mysql-connector-j</artifactId>
            <scope>runtime</scope>
        </dependency>

        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>

        <dependency>
            <!-- fastjson -->
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.47</version>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>


            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-resources-plugin</artifactId>
                <version>2.4.3</version>
            </plugin>

            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
                <version>2.22.2</version>
                <configuration>
                    <skipTests>true</skipTests>
                </configuration>
            </plugin>
        </plugins>
    </build>

</project>

 这样一来开发环境就搭建好了,项目结构如下所示

生产者

生产者主要就是使用web爬虫爬取csdn中的文章和作者信息,然后将消息推送到rocketmq中。

首先是配置文件

server.port=8081

rocketmq.name-server=192.168.232.129:9876
rocketmq.producer.group=CSDN_group

使用两个实体类来存储文章信息和作者信息

@Data
@AllArgsConstructor
@NoArgsConstructor
public class CSDNArticleEntity {

    private Long u_id;
    private String title;
    private String url;
    private String author_name;
    private String nick_name;
    private String up_time;
    private Integer read_volum;
    private Integer collection_volum;
    private String tag;
    private Double score;

}
@Data
@AllArgsConstructor
@NoArgsConstructor
public class CSDNAuthorEntity {

    private Long u_id;
    private String author_name;
    private String nick_name;
    private Integer num_fans;
    private Integer num_like;
    private Integer num_comment;
    private Integer num_collect;
    private Double author_score;

}

接着就是写爬虫,我们采用广度优先搜索的策略,以csdn的主页为源节点,加载文章url

(31条消息) CSDN博客-专业IT技术发表平台

加载的文章url都会存放到一个阻塞队列中。

然后再从阻塞队列中逐个将url取出,加载文章和作者信息,并且将页面中的推荐文章的url再放到阻塞队列中,这样就可以不停的爬文章信息了。

@Slf4j
@Service
public class CSDNSpiderServiceImpl implements CSDNSpiderService {

    @Autowired
    private RocketMQTemplate rocketMQTemplate;

    // 指定一个主题,两个TAG,分别对应文章消息和作者消息
    private final String TOPIC = "TOPIC_CSDN";
    private final String TAG1 = "ARTICLE";
    private final String TAG2 = "AUTHOR";

    private final String start_url = "https://blog.csdn.net/";
    // 定义一个线程池
    private ThreadPoolExecutor pool = new ThreadPoolExecutor(
            1,
            1,
            10,
            TimeUnit.SECONDS,
            new ArrayBlockingQueue<>(2048)
    );

    @Override
    public void startSpider() {
        new Thread(()->{
            // 首先从首页拿到初始的文章url
            Request request = new Request.Builder()
                    .url(start_url)
                    .get()   //默认就是GET请求,可以不写
                    .addHeader("User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.122 Safari/537.36")
                    .build();
            try {
                Response response = CSDNSpiderUtils.client.newCall(request).execute();
                Document dom = Jsoup.parse(response.body().string());
                // 找到文章
                Elements as = dom.getElementsByClass("blog");
                for(Element e : as) {
                    String url = e.attr("href");
                    if(url!=null && !url.equals("")) {
                        CSDNSpiderUtils.queue.offer(url);
                        log.debug("成功添加url:{}", url);
                    }
                }
                // 之后就是不断从队列中拿到url,然后解析顺便拿到更多的url
                while(true) {
                    // 如果队列为空就持续等待
                    while(CSDNSpiderUtils.queue.isEmpty()){}
                    String url = CSDNSpiderUtils.queue.poll();
                    pool.execute(new ArticleParser(url));
                    log.debug("成功消费url:{}", url);
                    Thread.sleep(500);
                }
            } catch (IOException e) {
                e.printStackTrace();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }).start();
    }


    // 解析每个页面的代码
    private class ArticleParser extends Thread {
        private String url = null;
        public ArticleParser(String url) {
            this.url = url;
        }

        @Override
        public void run() {
            super.run();
            // 判断一下url是否为空
            if(url==null || url.equals("")) {
                return;
            }
            // 先判断一下url是否是文章的url
            if(!url.contains("article")) {
                return;
            }
            // 然后再加载html页面然后解析
            // 建立连接
            Request request = new Request.Builder()
                    .url(url)
                    .get()   //默认就是GET请求,可以不写
                    .addHeader("User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.122 Safari/537.36")
                    .build();
            try {
                Response response = CSDNSpiderUtils.client.newCall(request).execute();
                Document dom = Jsoup.parse(response.body().string());
                // 获取作者信息
                CSDNAuthorEntity authorEntity = assembleAuthorEntity(dom, url);
                if(authorEntity == null) {
                    return;
                }
                // 获取文章信息
                CSDNArticleEntity articleEntity = assembleArticleEntity(dom, url, authorEntity.getAuthor_name(), authorEntity.getNick_name(), authorEntity.getAuthor_score());
                if(!CSDNSpiderUtils.scoreFilter(articleEntity, authorEntity)){
                    return;
                }

                /*
                * 向消息队列中推送消息
                * */
                if(articleEntity!=null) {
                    // 写入到消息队列
                    rocketMQTemplate.send(TOPIC + ":" + TAG1, MessageBuilder.withPayload(articleEntity).build());
                    log.info("获取文章信息:{}", JSON.toJSONString(articleEntity));
                }
                if(authorEntity != null) {
                    // 写入到消息队列
                    rocketMQTemplate.send(TOPIC + ":" + TAG2, MessageBuilder.withPayload(authorEntity).build());
                    log.info("获取作者信息:{}", JSON.toJSONString(authorEntity));
                }

                // 找到文章底部推荐的文章url
                Elements elementsByClass = dom.getElementsByClass("recommend-item-box type_blog clearfix");
                for(Element el : elementsByClass) {
                    String data_url = el.attr("data-url");
                    if(data_url!=null && !data_url.equals("")) {
                        CSDNSpiderUtils.queue.offer(data_url);
                        log.debug("成功添加url:{}", data_url);
                    }
                }
                elementsByClass = dom.getElementsByClass("recommend-item-box baiduSearch clearfix");
                for(Element el : elementsByClass) {
                    String data_url = el.attr("data-url");
                    if(data_url!=null && !data_url.equals("")) {
                        CSDNSpiderUtils.queue.offer(data_url);
                        log.debug("成功添加url:{}", data_url);
                    }
                }
                return;
            } catch (IOException e) {
                e.printStackTrace();
            }
            return;
        }
    }

    // 获取文章信息
    private CSDNArticleEntity assembleArticleEntity(Document dom, String url, String username, String nickname, double score) {
        // 找到文章标题
        String title = dom.getElementsByClass("title-article").text();
        // 如果title过长就截取
        if(title.length() >= 150) {
            title = title.substring(0, 150);
        }
        // 找到发表时间
        String up_time = "";
        if(dom.getElementsByClass("time").text().split(" ").length>=2) {
            up_time = dom.getElementsByClass("time").text().split(" ")[1];
        }
        else {
            return null;
        }
        // 找到文章阅读量
        if(dom.getElementsByClass("read-count").text()==null || dom.getElementsByClass("read-count").text().equals("")){
            return null;
        }
        Integer read_volum = Integer.parseInt(dom.getElementsByClass("read-count").text());
        // 找到文章收藏量
        if(dom.getElementsByClass("get-collection").text()==null || dom.getElementsByClass("get-collection").text().equals("")){
            return null;
        }
        Integer collection_volum = Integer.parseInt(dom.getElementsByClass("get-collection").text().split(" ")[0]);

        // 查找文章标签
        StringBuilder sb = new StringBuilder();
        Element element = dom.getElementsByClass("tags-box artic-tag-box").first();
        // 找到所有的a标签
        // 第一个标签是文章分类的,所以忽略掉
        // 第二个是个span,内容是“文章标签”,也忽略掉
        Elements tag_as = element.getElementsByTag("a").next().next();
        for(Element tag_a : tag_as) {
            sb.append(tag_a.text());
            sb.append(" ");
        }
        String str_tag = sb.toString();
        CSDNArticleEntity articleEntity = new CSDNArticleEntity();
        articleEntity.setTitle(title);
        articleEntity.setUp_time(up_time);
        articleEntity.setAuthor_name(username);
        articleEntity.setNick_name(nickname);
        articleEntity.setRead_volum(read_volum);
        articleEntity.setCollection_volum(collection_volum);
        articleEntity.setTag(str_tag);
        articleEntity.setUrl(url);
        articleEntity.setScore(CSDNSpiderUtils.getScore(read_volum, collection_volum, up_time, score));
        return articleEntity;
    }

    // 获取作者信息
    private CSDNAuthorEntity assembleAuthorEntity(Document dom, String url) {
        CSDNAuthorEntity authorEntity = new CSDNAuthorEntity();
        // 从url中获取作者username
        String username = url.split("/article")[0].split("https://blog.csdn.net/")[1];
        authorEntity.setAuthor_name(username);
        // 获取作者的昵称
        Element name_el = dom.getElementsByClass("profile-intro-name-boxTop").first();
        String nick_name = name_el.getElementsByTag("a").first().getElementsByTag("span").text();
        authorEntity.setNick_name(nick_name);
        Element element = dom.getElementsByClass("data-info d-flex item-tiling").get(1);
        Elements elements = element.getElementsByTag("dl");
        try {
            Integer num_fans = Integer.parseInt(elements.get(1).attr("title"));
            Integer num_like = Integer.parseInt(elements.get(2).attr("title"));
            Integer num_comment = Integer.parseInt(elements.get(3).attr("title"));
            Integer num_collect = Integer.parseInt(elements.get(4).attr("title"));
            authorEntity.setNum_fans(num_fans);
            authorEntity.setNum_like(num_like);
            authorEntity.setNum_comment(num_comment);
            authorEntity.setNum_collect(num_collect);
            authorEntity.setAuthor_score(CSDNSpiderUtils.getAuthorScore(num_fans, num_like, num_comment, num_collect));
        } catch (Exception e) {
            log.debug(elements.get(1).attr("title"));
            log.debug("出错的url:{}", url);
        }
        return authorEntity;
    }

}

从上述代码中可以看出来,我们定义了一个TOPIC和两个TAG,分别用来存放文章信息和作者信息,我们启动服务可以发现消息队列中已经有消息了。

可以在web控制台中看到生产者信息

消费者

消费者负责从rocketmq中取出消息,并且将数据存到mysql数据库中。

首先是配置信息,分别是mysql的配置,mybatis的配置和rocketmq的配置,具体的配置信息你需要根据实际情况来填写

server.port=8082

# mysql
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
spring.datasource.url=jdbc:mysql://192.168.232.129:3306/rocketmq_demo?serverTimezone=UTC&&characterEncoding=UTF-8
spring.datasource.username=root
spring.datasource.password=123456
# mybatis
mybatis.type-aliases-package=com.example.demo.entity
mybatis.mapper-locations=classpath:/mapper/*Mapper.xml
# rocketmq
rocketmq.name-server=192.168.232.129:9876

然后是建库建表,我建的数据库名称是rocketmq_demo,你也可以用其他名称,建表sql如下所示

USE rocketmq_demo;
-- ----------------------------
-- Table structure for csdn_article
-- ----------------------------
DROP TABLE IF EXISTS `csdn_article`;
CREATE TABLE `csdn_article`  (
  `u_id` bigint(20) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '文章id',
  `title` varchar(150) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '文章标题',
  `url` varchar(200) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '文章url',
  `author_name` varchar(50) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '文章作者',
  `nick_name` varchar(50) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '作者昵称',
  `up_time` varchar(10) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '文章发表时间 yyyy-mm-dd',
  `read_volum` int(11) NOT NULL COMMENT '文章阅读量',
  `collection_volum` int(11) NOT NULL COMMENT '文章收藏量',
  `tag` varchar(100) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '文章标签',
  `article_score` double NOT NULL COMMENT '文章得分',
  PRIMARY KEY (`u_id`) USING BTREE,
  UNIQUE INDEX `url_index`(`url`) USING BTREE,
  INDEX `title_index`(`title`) USING BTREE,
  INDEX `tag_index`(`tag`) USING BTREE,
  INDEX `score_index`(`article_score`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 2615 CHARACTER SET = utf8 COLLATE = utf8_general_ci COMMENT = '文章信息表' ROW_FORMAT = Dynamic;

-- ----------------------------
-- Table structure for csdn_author
-- ----------------------------
DROP TABLE IF EXISTS `csdn_author`;
CREATE TABLE `csdn_author`  (
  `u_id` bigint(20) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '作者id',
  `author_name` varchar(50) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '文章作者',
  `nick_name` varchar(50) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '作者昵称',
  `num_fans` int(11) NOT NULL COMMENT '粉丝数量',
  `num_like` int(11) NOT NULL COMMENT '获赞数量',
  `num_comment` int(11) NOT NULL COMMENT '评论数量',
  `num_collect` int(11) NOT NULL COMMENT '收藏数量',
  `author_score` double NOT NULL COMMENT '作者得分',
  PRIMARY KEY (`u_id`) USING BTREE,
  UNIQUE INDEX `author_name_index`(`author_name`) USING BTREE,
  INDEX `author_socre_index`(`author_score`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 2306 CHARACTER SET = utf8 COLLATE = utf8_general_ci COMMENT = '作者信息表' ROW_FORMAT = Dynamic;

SET FOREIGN_KEY_CHECKS = 1;

我们需要监听同一个TOPIC的两个TAG,这边需要注意一下,不同TAG的消费组需要不同,不然会报错。这个问题我真的找了好久好久,最后才发现是消费组相同造成的,-_-||

@Slf4j
@Service
@RocketMQMessageListener(topic = "TOPIC_CSDN", selectorExpression = "ARTICLE", consumerGroup = "CSDN_ARTICLE")
public class ArticleListener implements RocketMQListener<CSDNArticleEntity> {
    @Autowired
    private CSDNMapper csdnMapper;

    @Override
    public void onMessage(CSDNArticleEntity csdnArticleEntity) {
        log.info("消费文章信息:{}", csdnArticleEntity);
        if(csdnMapper.checkArticle(csdnArticleEntity)==0) {
            csdnMapper.insertOneArticle(csdnArticleEntity);
        }
        else {
            csdnMapper.updateOneArticle(csdnArticleEntity);
        }
    }
}
@Slf4j
@Service
@RocketMQMessageListener(topic = "TOPIC_CSDN", selectorExpression = "AUTHOR", consumerGroup = "CSDN_AUTHOR")
public class AuthorListener implements RocketMQListener<CSDNAuthorEntity> {

    @Autowired
    private CSDNMapper csdnMapper;

    @Override

    public void onMessage(CSDNAuthorEntity csdnAuthorEntity) {
        log.info("消费作者信息:{}", csdnAuthorEntity);
        if(csdnMapper.checkAuthor(csdnAuthorEntity)==0) {
            csdnMapper.insertOneAuthor(csdnAuthorEntity);
        }
        else {
            csdnMapper.updateOneAuthor(csdnAuthorEntity);
        }
    }
}

可以看到这其中有数据库操作,对应的数据库mapper如下

@Mapper
public interface CSDNMapper {

    // 清空数据库
    public void clearArticle();
    public void clearAuthor();

    // 检查是否有某篇文章
    public int checkArticle(CSDNArticleEntity articleEntity);

    // 检查是否有某个作者
    public int checkAuthor(CSDNAuthorEntity authorEntity);

    // 插入一条数据
    public void insertOneArticle(CSDNArticleEntity articleEntity);

    // 插入一条作者数据
    public void insertOneAuthor(CSDNAuthorEntity authorEntity);

    // 更新一条数据
    public void updateOneArticle(CSDNArticleEntity articleEntity);

    public void updateOneAuthor(CSDNAuthorEntity authorEntity);

    // 从文章表中查询数据
    public List<CSDNArticleEntity> selectFromArticle(@Param("keyword") String keyword,
                                                     @Param("offset") Integer offset,
                                                     @Param("pagesize") Integer pagesize);
    // 查询一共有多少条数据
    public Integer selectCount(@Param("keyword") String keyword);

}
<?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.example.demo.mapper.CSDNMapper">

    <delete id="clearArticle">
        delete from csdn_article where 1=1
    </delete>
    <delete id="clearAuthor">
        delete from csdn_author where 1=1
    </delete>

    <select id="checkArticle" parameterType="CSDNArticleEntity" resultType="Integer">
        select count(url) from csdn_article
        where url=#{url}
    </select>

    <select id="checkAuthor" parameterType="CSDNAuthorEntity" resultType="Integer">
        select count(author_name) from csdn_author
        where author_name=#{author_name}
    </select>

    <insert id="insertOneArticle" parameterType="CSDNArticleEntity">
        insert into csdn_article
        (title, url, up_time, author_name, nick_name, read_volum, collection_volum, tag, article_score) values
        (#{title}, #{url}, #{up_time}, #{author_name}, #{nick_name}, #{read_volum}, #{collection_volum}, #{tag}, #{score})
    </insert>

    <insert id="insertOneAuthor" parameterType="CSDNAuthorEntity">
        insert into csdn_author
        (author_name, nick_name, num_fans, num_like, num_comment, num_collect, author_score) values
        (#{author_name}, #{nick_name}, #{num_fans}, #{num_like}, #{num_comment}, #{num_collect}, #{author_score})
    </insert>

    <update id="updateOneArticle" parameterType="CSDNArticleEntity">
        update csdn_article
        set
        read_volum = #{read_volum},
        collection_volum = #{collection_volum},
        article_score = #{score}
        where url=#{url}
    </update>

    <update id="updateOneAuthor" parameterType="CSDNAuthorEntity">
        update csdn_author
        set
        num_fans=#{num_fans},
        num_like=#{num_like},
        num_comment=#{num_comment},
        num_collect=#{num_collect},
        author_score=#{author_score}
        where author_name=#{author_name}
    </update>

    <select id="selectFromArticle" resultType="CSDNArticleEntity">
        select * from csdn_article
        where
        title like CONCAT('%',#{keyword},'%') or
        tag like CONCAT('%',#{keyword},'%')
        order by article_score DESC
        limit #{offset}, #{pagesize}
    </select>

    <select id="selectCount" parameterType="String" resultType="Integer">
        select count(u_id) from csdn_article
        where
        title like CONCAT('%',#{keyword},'%') or
        tag like CONCAT('%',#{keyword},'%')
    </select>


</mapper>

将消费者启动起来,可以在web控制台中查看到消费者信息

查看数据库,发现有数据了,消费成功

总结

这个项目虽然看上去不难,但是真的做起来还是花了不少精力的。

我将代码都放到gitee上了,有需要可以自取。

rocketmq-demo: springboot整合rocketmq的一个demo (gitee.com)

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

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

相关文章

CSDN 周赛填空题,充满恶意的嘲讽

CSDN 周赛填空题&#xff0c;充满恶意的嘲讽41期的填空题44期的填空题45期的填空题再说题型老顾最近一直在玩 csdn 周赛&#xff0c;没啥想法&#xff0c;就是想票点小玩意&#xff0c;之前从第四十一期开始&#xff0c;题型进行了扩展&#xff0c;增加了填空、判断、单选。扩展…

C#调试与测试 | Assert(断言)

Assert(断言) 文章目录Assert(断言)前言什么是Assert适用场景使用示例检查传入的参数是否为空检查循环变量是否在规定范围内检查方法返回值是否为null结束语前言 今天我要和大家聊聊C#调试和测试中的一种神器——断言&#xff08;Assert&#xff09;。如果你还不知道什么是断言…

nvm-windows的安装使用及踩坑指南

nvm是node的一款版本管理工具&#xff0c;可以简单操作node版本的切换、安装、查看。常规来说&#xff0c;开发中安装一个node版本就够了&#xff0c;但是最近在开发中有的老项目或者一些特定的项目需要来回切换node的版本&#xff0c;不可能手动去卸载掉之前的node版本&#x…

轻松掌握微服务治理的注册中心Eureka到Nacos知识点

1、SpringCloud 1、介绍 2、消费者与服务者 3、服务拆分 1、介绍 2、服务之间调用 例如有两个微服务&#xff0c;分别提供用户信息和订单信息。两个服务都有自己的数据库&#xff0c;所以如下查订单信息是不能直接去查用户信息的数据库的&#xff0c;只能从订单服务发起远程…

Seata 将参展 SOFA 五周年开源集市~

SOFA 五周年开源集市4 月 15 日&#xff08;本周六&#xff09;&#xff0c;SOFAStack 社区将在北京市朝阳区恒通国际创新园 C work 举办开源五周年活动&#xff0c;现场将于 12:00 正式开放开源集市&#xff0c;欢迎感兴趣的开发者们前来参与。今天让我们一起走近认识下开源的…

上海雷卯推出DFN1006超小体积网口保护ESD ULC0342P26LV 带回扫

什么是静电放电&#xff1f; 静电放电是指由于两种不同材料之间的摩擦或分离而产生的电荷累积所引起的电荷释放现象。 为什么需要防静电保护&#xff1f; 静电放电会对电子设备造成损害&#xff0c;特别是对于灵敏的网络设备来说&#xff0c;静电放电可能导致设备损坏或失效。…

图卷积网络GCN---底层逻辑最简单直白的理解

一 、GNN是怎么被提出来的&#xff1f; 比较常见的有CNN、RNN等。CNN的核心在于它的kernel&#xff0c;kernel在图片上平移&#xff0c;通过卷积的方式来提取特征。这里的关键在于图片结构上的平移不变性&#xff1a;一个小窗口无论移动到图片的哪一个位置&#xff0c;其内部的…

Elk运维-elastic7.6.1集群安装部署

wei集群安装结果说明 实例配置安装软件安装账号hadoop1 2C4G 磁盘&#xff1a;50G 云服务器 elasticsearch kibana rdhadoop2 2C4G 磁盘&#xff1a;50G 云服务器 elasticsearchrdhadoop3 2C4G 磁盘&#xff1a;50G 云服务器 elasticsearchrd整个安装过程使用的账号&…

linux(docker)下使用VuePress从零开始搭建自己的博客(一):VuePress环境搭建

目标&#xff1a;在linux centos下使用VuePress搭建博客系统&#xff0c;最好能利用docker进行备份和迁移。 本节内容&#xff1a;本节主要讲述VuePress的环境搭建过程&#xff0c;VuePress的基本配置以及备份和迁移。详细记录了从零开始搭建的过程&#xff0c;以及搭建过程中遇…

Apache配置与应用

1.基于域名的虚拟主机 为虚拟主机提供域名解析 基于域名&#xff1a;为每个虚拟主机使用不同的域名&#xff0c;但是其对应的 IP 地址是相同的。例如&#xff0c;www.benet.com 和 www.accp.com 站点的 IP 地址都是 192.168.80.10。这是使用最为普遍的虚拟 Web 主机类型。 方法…

【Java Web】015 -- Maven高级(分模块设计与开发、继承与聚合、私服)

目录 一、分模块设计与开发 1、为什么要分模块设计&#xff1f; 2、实践&#xff1a;分模块开发 ①、实现步骤 3、小结 二、继承与聚合 继承 1、继承关系 ①、为什么要在Maven工程中实现继承&#xff1f; ②、继承关系实现 ③、继承实现小结 ④、maven项目父子工程结构说明 2、…

Java Stream API 操作完全攻略:让你的代码更加出色 (二)

前言 Java Stream 是一种强大的数据处理工具&#xff0c;可以帮助开发人员快速高效地处理和转换数据流。使用 Stream 操作可以大大简化代码&#xff0c;使其更具可读性和可维护性&#xff0c;从而提高开发效率。本文将为您介绍 Java Stream 操作的所有方面&#xff0c;包括 red…

WMS仓库管理系统,你能操作明白吗?别让"智能"变"滞能"

随着社会的进步&#xff0c;智能车间、智能工厂等申报的展开&#xff0c;“智能化”的概念&#xff0c;让企业系统出现更迭。以智能化仓储管理系统为例&#xff0c;企业工厂利用WMS的优势&#xff0c;依照运行的工作标准和运算法则&#xff0c;对仓库进行精细化管理。 WM…

IC学习笔记22——memory_compilermemory_wrapper

一、memory_compiler 1.1 memory_compiler的介绍 memory_compiler为一系列工具的统称,用于生成芯片开发所需要的memory。芯片开发中所需要的memory为sram、rom等。很多公司都有自己开发的memory_compiler工具。 1.2 SRAM基础 sram写操作(写1) 将要写入的数据“1”通过写入…

Vue学习笔记(3. 基本语法,数据绑定,指令)

1. 基本语法 vue2的基本语法如下&#xff1a; <html> <head> <title>testVUE</title> <script src"https://cdn.staticfile.org/vue/2.2.2/vue.min.js"></script> </head><body><div id"app">模板…

【操作系统】深入理解Linux虚拟内存管理

【操作系统】深入理解Linux虚拟内存管理 本篇跟大家说说内存管理&#xff0c;内存管理还是比较重要的一个环节&#xff0c;理解了它&#xff0c;至少对整个操作系统的工作会有一个初步的轮廓。 虚拟内存 如果你是电子相关专业的&#xff0c;肯定在大学里捣鼓过单片机。 单片…

出口欧盟认证机构为什么推荐CE认证

产品做CE证书要什么好处&#xff0c;CE证书要什么作用&#xff1f; 在中国很多中小企业困扰的是&#xff0c;好不容易走出国门接到来自国外的订单。却因为没有相应的产品安全认证而无法顺利取得订单。或者当产品到达目的地后&#xff0c;因为无法提供相应的产品安全认证证明文件…

网络编程 端口状态都有哪些?

客户端各个状态&#xff1a; 三次握手阶段 CLOSED&#xff08;持续状态&#xff09; 关闭状态 SYN_SENT &#xff08;非常短暂&#xff09; SYN_SENT状态表示请求连接&#xff0c;当你要访问其它的计算机的服务时首先要发个同步信号给该端口&#xff0c;此时状态为SYN_SENT&a…

数据库MySQL —— 存储过程

目录 一、介绍 二、基本语法 三、变量 1. 系统变量 2. 用户自定义变量 3. 局部变量 四、流程控制语句 1. if判断 2. 参数 3. case 4. 循环 4.1 while 4.2 repeat 4.3 loop 五、游标 - cursor 六、条件处理程序 - handler 七、存储函数 一、介绍 存储过…

自学大数据第17天~Hive安装与配置及常用命令

嗨咯,好几天没更新了,你以为我不知道吗? 这几天主要还是学习,学习如何再学习! 简述以下这几天都学习了什么~ MongoDB的命令 GaussDB(for Mongo)集群架构与社区版架构的对比 社区版(副本集架构): mongos(路由) config(记录元数据) shard(负责计算和存储) GaussDB(for Mongo)存…