点赞模块设计 - Redis缓存 + 定时写入数据库实现高性能点赞功能

点赞模块设计 - Redis缓存 + 定时写入数据库实现高性能点赞功能

源码地址:github.com/cachecats/c…

点赞是作为整个系统的一个小模块,代码在 user-service 用户服务下。


本文基于 SpringCloud, 用户发起点赞、取消点赞后先存入 Redis 中,再每隔两小时从 Redis 读取点赞数据写入数据库中做持久化存储。

点赞功能在很多系统中都有,但别看功能小,想要做好需要考虑的东西还挺多的。

点赞、取消点赞是高频次的操作,若每次都读写数据库,大量的操作会影响数据库性能,所以需要做缓存。

至于多久从 Redis 取一次数据存到数据库中,根据项目的实际情况定吧,我是暂时设了两个小时。

项目需求需要查看都谁点赞了,所以要存储每个点赞的点赞人、被点赞人,不能简单的做计数。

文章分四部分介绍:

  • Redis 缓存设计及实现
  • 数据库设计
  • 数据库操作
  • 开启定时任务持久化存储到数据库

一、Redis 缓存设计及实现

1.1 Redis 安装及运行

Redis 安装请自行查阅相关教程。

说下Docker 安装运行 Redis

复制代码docker run -d -p 6379:6379 redis:4.0.8

如果已经安装了 Redis,打开命令行,输入启动 Redis 的命令

复制代码redis-server

1.2 Redis 与 SpringBoot 项目的整合

  1. pom.xml 中引入依赖
复制代码<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
  1. 在启动类上添加注释 @EnableCaching
复制代码@SpringBootApplication
@EnableDiscoveryClient
@EnableSwagger2
@EnableFeignClients(basePackages = "com.solo.coderiver.project.client")
@EnableCaching
public class UserApplication {public static void main(String[] args) {SpringApplication.run(UserApplication.class, args);}
}
  1. 编写 Redis 配置类 RedisConfig
复制代码import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;import java.net.UnknownHostException;@Configuration
public class RedisConfig {@Bean@ConditionalOnMissingBean(name = "redisTemplate")public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory)throws UnknownHostException {Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<Object>(Object.class);ObjectMapper om = new ObjectMapper();om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);jackson2JsonRedisSerializer.setObjectMapper(om);RedisTemplate<String, Object> template = new RedisTemplate<String, Object>();template.setConnectionFactory(redisConnectionFactory);template.setKeySerializer(jackson2JsonRedisSerializer);template.setValueSerializer(jackson2JsonRedisSerializer);template.setHashKeySerializer(jackson2JsonRedisSerializer);template.setHashValueSerializer(jackson2JsonRedisSerializer);template.afterPropertiesSet();return template;}@Bean@ConditionalOnMissingBean(StringRedisTemplate.class)public StringRedisTemplate stringRedisTemplate(RedisConnectionFactory redisConnectionFactory)throws UnknownHostException {StringRedisTemplate template = new StringRedisTemplate();template.setConnectionFactory(redisConnectionFactory);return template;}
}

至此 Redis 在 SpringBoot 项目中的配置已经完成,可以愉快的使用了。

1.3 Redis 的数据结构类型

Redis 可以存储键与5种不同数据结构类型之间的映射,这5种数据结构类型分别为String(字符串)、List(列表)、Set(集合)、Hash(散列)和 Zset(有序集合)。

下面来对这5种数据结构类型作简单的介绍:

结构类型结构存储的值结构的读写能力
String可以是字符串、整数或者浮点数对整个字符串或者字符串的其中一部分执行操作;对象和浮点数执行自增(increment)或者自减(decrement)
List一个链表,链表上的每个节点都包含了一个字符串从链表的两端推入或者弹出元素;根据偏移量对链表进行修剪(trim);读取单个或者多个元素;根据值来查找或者移除元素
Set包含字符串的无序收集器(unorderedcollection),并且被包含的每个字符串都是独一无二的、各不相同添加、获取、移除单个元素;检查一个元素是否存在于某个集合中;计算交集、并集、差集;从集合里卖弄随机获取元素
Hash包含键值对的无序散列表添加、获取、移除单个键值对;获取所有键值对
Zset字符串成员(member)与浮点数分值(score)之间的有序映射,元素的排列顺序由分值的大小决定添加、获取、删除单个元素;根据分值范围(range)或者成员来获取元素

1.4 点赞数据在 Redis 中的存储格式

用 Redis 存储两种数据,一种是记录点赞人、被点赞人、点赞状态的数据,另一种是每个用户被点赞了多少次,做个简单的计数。

由于需要记录点赞人和被点赞人,还有点赞状态(点赞、取消点赞),还要固定时间间隔取出 Redis 中所有点赞数据,分析了下 Redis 数据格式中 Hash 最合适。

因为 Hash 里的数据都是存在一个键里,可以通过这个键很方便的把所有的点赞数据都取出。这个键里面的数据还可以存成键值对的形式,方便存入点赞人、被点赞人和点赞状态。

设点赞人的 id 为 likedPostId,被点赞人的 id 为 likedUserId ,点赞时状态为 1,取消点赞状态为 0。将点赞人 id 和被点赞人 id 作为键,两个 id 中间用 :: 隔开,点赞状态作为值。

所以如果用户点赞,存储的键为:likedUserId::likedPostId,对应的值为 1 。

取消点赞,存储的键为:likedUserId::likedPostId,对应的值为 0 。

取数据时把键用 :: 切开就得到了两个id,也很方便。

在可视化工具 RDM 中看到的是这样子

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

img

1.5 操作 Redis

Redis 各种数据格式的操作方法可以看看 这篇文章 ,写的非常好。

将具体操作方法封装到了 RedisService 接口里

RedisService.java

复制代码import com.solo.coderiver.user.dataobject.UserLike;
import com.solo.coderiver.user.dto.LikedCountDTO;import java.util.List;public interface RedisService {/*** 点赞。状态为1* @param likedUserId* @param likedPostId*/void saveLiked2Redis(String likedUserId, String likedPostId);/*** 取消点赞。将状态改变为0* @param likedUserId* @param likedPostId*/void unlikeFromRedis(String likedUserId, String likedPostId);/*** 从Redis中删除一条点赞数据* @param likedUserId* @param likedPostId*/void deleteLikedFromRedis(String likedUserId, String likedPostId);/*** 该用户的点赞数加1* @param likedUserId*/void incrementLikedCount(String likedUserId);/*** 该用户的点赞数减1* @param likedUserId*/void decrementLikedCount(String likedUserId);/*** 获取Redis中存储的所有点赞数据* @return*/List<UserLike> getLikedDataFromRedis();/*** 获取Redis中存储的所有点赞数量* @return*/List<LikedCountDTO> getLikedCountFromRedis();}

实现类 RedisServiceImpl.java

复制代码import com.solo.coderiver.user.dataobject.UserLike;
import com.solo.coderiver.user.dto.LikedCountDTO;
import com.solo.coderiver.user.enums.LikedStatusEnum;
import com.solo.coderiver.user.service.LikedService;
import com.solo.coderiver.user.service.RedisService;
import com.solo.coderiver.user.utils.RedisKeyUtils;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.Cursor;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.ScanOptions;
import org.springframework.stereotype.Service;import java.util.ArrayList;
import java.util.List;
import java.util.Map;@Service
@Slf4j
public class RedisServiceImpl implements RedisService {@AutowiredRedisTemplate redisTemplate;@AutowiredLikedService likedService;@Overridepublic void saveLiked2Redis(String likedUserId, String likedPostId) {String key = RedisKeyUtils.getLikedKey(likedUserId, likedPostId);redisTemplate.opsForHash().put(RedisKeyUtils.MAP_KEY_USER_LIKED, key, LikedStatusEnum.LIKE.getCode());}@Overridepublic void unlikeFromRedis(String likedUserId, String likedPostId) {String key = RedisKeyUtils.getLikedKey(likedUserId, likedPostId);redisTemplate.opsForHash().put(RedisKeyUtils.MAP_KEY_USER_LIKED, key, LikedStatusEnum.UNLIKE.getCode());}@Overridepublic void deleteLikedFromRedis(String likedUserId, String likedPostId) {String key = RedisKeyUtils.getLikedKey(likedUserId, likedPostId);redisTemplate.opsForHash().delete(RedisKeyUtils.MAP_KEY_USER_LIKED, key);}@Overridepublic void incrementLikedCount(String likedUserId) {redisTemplate.opsForHash().increment(RedisKeyUtils.MAP_KEY_USER_LIKED_COUNT, likedUserId, 1);}@Overridepublic void decrementLikedCount(String likedUserId) {redisTemplate.opsForHash().increment(RedisKeyUtils.MAP_KEY_USER_LIKED_COUNT, likedUserId, -1);}@Overridepublic List<UserLike> getLikedDataFromRedis() {Cursor<Map.Entry<Object, Object>> cursor = redisTemplate.opsForHash().scan(RedisKeyUtils.MAP_KEY_USER_LIKED, ScanOptions.NONE);List<UserLike> list = new ArrayList<>();while (cursor.hasNext()){Map.Entry<Object, Object> entry = cursor.next();String key = (String) entry.getKey();//分离出 likedUserId,likedPostIdString[] split = key.split("::");String likedUserId = split[0];String likedPostId = split[1];Integer value = (Integer) entry.getValue();//组装成 UserLike 对象UserLike userLike = new UserLike(likedUserId, likedPostId, value);list.add(userLike);//存到 list 后从 Redis 中删除redisTemplate.opsForHash().delete(RedisKeyUtils.MAP_KEY_USER_LIKED, key);}return list;}@Overridepublic List<LikedCountDTO> getLikedCountFromRedis() {Cursor<Map.Entry<Object, Object>> cursor = redisTemplate.opsForHash().scan(RedisKeyUtils.MAP_KEY_USER_LIKED_COUNT, ScanOptions.NONE);List<LikedCountDTO> list = new ArrayList<>();while (cursor.hasNext()){Map.Entry<Object, Object> map = cursor.next();//将点赞数量存储在 LikedCountDTString key = (String)map.getKey();LikedCountDTO dto = new LikedCountDTO(key, (Integer) map.getValue());list.add(dto);//从Redis中删除这条记录redisTemplate.opsForHash().delete(RedisKeyUtils.MAP_KEY_USER_LIKED_COUNT, key);}return list;}
}

用到的工具类和枚举类

RedisKeyUtils, 用于根据一定规则生成 key

复制代码public class RedisKeyUtils {//保存用户点赞数据的keypublic static final String MAP_KEY_USER_LIKED = "MAP_USER_LIKED";//保存用户被点赞数量的keypublic static final String MAP_KEY_USER_LIKED_COUNT = "MAP_USER_LIKED_COUNT";/*** 拼接被点赞的用户id和点赞的人的id作为key。格式 222222::333333* @param likedUserId 被点赞的人id* @param likedPostId 点赞的人的id* @return*/public static String getLikedKey(String likedUserId, String likedPostId){StringBuilder builder = new StringBuilder();builder.append(likedUserId);builder.append("::");builder.append(likedPostId);return builder.toString();}
}

LikedStatusEnum 用户点赞状态的枚举类

复制代码package com.solo.coderiver.user.enums;import lombok.Getter;/*** 用户点赞的状态*/
@Getter
public enum LikedStatusEnum {LIKE(1, "点赞"),UNLIKE(0, "取消点赞/未点赞"),;private Integer code;private String msg;LikedStatusEnum(Integer code, String msg) {this.code = code;this.msg = msg;}
}

二、数据库设计

数据库表中至少要包含三个字段:被点赞用户id,点赞用户id,点赞状态。再加上主键id,创建时间,修改时间就行了。

建表语句

复制代码create table `user_like`(`id` int not null auto_increment,`liked_user_id` varchar(32) not null comment '被点赞的用户id',`liked_post_id` varchar(32) not null comment '点赞的用户id',`status` tinyint(1) default '1' comment '点赞状态,0取消,1点赞',`create_time` timestamp not null default current_timestamp comment '创建时间',`update_time` timestamp not null default current_timestamp on update current_timestamp comment '修改时间',primary key(`id`),INDEX `liked_user_id`(`liked_user_id`),INDEX `liked_post_id`(`liked_post_id`)
) comment '用户点赞表';

对应的对象 UserLike

复制代码import com.solo.coderiver.user.enums.LikedStatusEnum;
import lombok.Data;import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;/*** 用户点赞表*/
@Entity
@Data
public class UserLike {//主键id@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Integer id;//被点赞的用户的idprivate String likedUserId;//点赞的用户的idprivate String likedPostId;//点赞的状态.默认未点赞private Integer status = LikedStatusEnum.UNLIKE.getCode();public UserLike() {}public UserLike(String likedUserId, String likedPostId, Integer status) {this.likedUserId = likedUserId;this.likedPostId = likedPostId;this.status = status;}
}

三、数据库操作

操作数据库同样封装在接口中

LikedService

复制代码import com.solo.coderiver.user.dataobject.UserLike;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;import java.util.List;public interface LikedService {/*** 保存点赞记录* @param userLike* @return*/UserLike save(UserLike userLike);/*** 批量保存或修改* @param list*/List<UserLike> saveAll(List<UserLike> list);/*** 根据被点赞人的id查询点赞列表(即查询都谁给这个人点赞过)* @param likedUserId 被点赞人的id* @param pageable* @return*/Page<UserLike> getLikedListByLikedUserId(String likedUserId, Pageable pageable);/*** 根据点赞人的id查询点赞列表(即查询这个人都给谁点赞过)* @param likedPostId* @param pageable* @return*/Page<UserLike> getLikedListByLikedPostId(String likedPostId, Pageable pageable);/*** 通过被点赞人和点赞人id查询是否存在点赞记录* @param likedUserId* @param likedPostId* @return*/UserLike getByLikedUserIdAndLikedPostId(String likedUserId, String likedPostId);/*** 将Redis里的点赞数据存入数据库中*/void transLikedFromRedis2DB();/*** 将Redis中的点赞数量数据存入数据库*/void transLikedCountFromRedis2DB();}

LikedServiceImpl 实现类

复制代码import com.solo.coderiver.user.dataobject.UserInfo;
import com.solo.coderiver.user.dataobject.UserLike;
import com.solo.coderiver.user.dto.LikedCountDTO;
import com.solo.coderiver.user.enums.LikedStatusEnum;
import com.solo.coderiver.user.repository.UserLikeRepository;
import com.solo.coderiver.user.service.LikedService;
import com.solo.coderiver.user.service.RedisService;
import com.solo.coderiver.user.service.UserService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;import java.util.List;@Service
@Slf4j
public class LikedServiceImpl implements LikedService {@AutowiredUserLikeRepository likeRepository;@AutowiredRedisService redisService;@AutowiredUserService userService;@Override@Transactionalpublic UserLike save(UserLike userLike) {return likeRepository.save(userLike);}@Override@Transactionalpublic List<UserLike> saveAll(List<UserLike> list) {return likeRepository.saveAll(list);}@Overridepublic Page<UserLike> getLikedListByLikedUserId(String likedUserId, Pageable pageable) {return likeRepository.findByLikedUserIdAndStatus(likedUserId, LikedStatusEnum.LIKE.getCode(), pageable);}@Overridepublic Page<UserLike> getLikedListByLikedPostId(String likedPostId, Pageable pageable) {return likeRepository.findByLikedPostIdAndStatus(likedPostId, LikedStatusEnum.LIKE.getCode(), pageable);}@Overridepublic UserLike getByLikedUserIdAndLikedPostId(String likedUserId, String likedPostId) {return likeRepository.findByLikedUserIdAndLikedPostId(likedUserId, likedPostId);}@Override@Transactionalpublic void transLikedFromRedis2DB() {List<UserLike> list = redisService.getLikedDataFromRedis();for (UserLike like : list) {UserLike ul = getByLikedUserIdAndLikedPostId(like.getLikedUserId(), like.getLikedPostId());if (ul == null){//没有记录,直接存入save(like);}else{//有记录,需要更新ul.setStatus(like.getStatus());save(ul);}}}@Override@Transactionalpublic void transLikedCountFromRedis2DB() {List<LikedCountDTO> list = redisService.getLikedCountFromRedis();for (LikedCountDTO dto : list) {UserInfo user = userService.findById(dto.getId());//点赞数量属于无关紧要的操作,出错无需抛异常if (user != null){Integer likeNum = user.getLikeNum() + dto.getCount();user.setLikeNum(likeNum);//更新点赞数量userService.updateInfo(user);}}}
}

数据库的操作就这些,主要还是增删改查。

四、开启定时任务持久化存储到数据库

定时任务 Quartz 很强大,就用它了。

Quartz 使用步骤:

  1. 添加依赖
复制代码<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-quartz</artifactId>
</dependency>
  1. 编写配置文件
复制代码package com.solo.coderiver.user.config;import com.solo.coderiver.user.task.LikeTask;
import org.quartz.*;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;@Configuration
public class QuartzConfig {private static final String LIKE_TASK_IDENTITY = "LikeTaskQuartz";@Beanpublic JobDetail quartzDetail(){return JobBuilder.newJob(LikeTask.class).withIdentity(LIKE_TASK_IDENTITY).storeDurably().build();}@Beanpublic Trigger quartzTrigger(){SimpleScheduleBuilder scheduleBuilder = SimpleScheduleBuilder.simpleSchedule()
//                .withIntervalInSeconds(10)  //设置时间周期单位秒.withIntervalInHours(2)  //两个小时执行一次.repeatForever();return TriggerBuilder.newTrigger().forJob(quartzDetail()).withIdentity(LIKE_TASK_IDENTITY).withSchedule(scheduleBuilder).build();}
}
  1. 编写执行任务的类继承自 QuartzJobBean
复制代码package com.solo.coderiver.user.task;import com.solo.coderiver.user.service.LikedService;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang.time.DateUtils;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.quartz.QuartzJobBean;import java.text.SimpleDateFormat;
import java.util.Date;/*** 点赞的定时任务*/
@Slf4j
public class LikeTask extends QuartzJobBean {@AutowiredLikedService likedService;private SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");@Overrideprotected void executeInternal(JobExecutionContext jobExecutionContext) throws JobExecutionException {log.info("LikeTask-------- {}", sdf.format(new Date()));//将 Redis 里的点赞信息同步到数据库里likedService.transLikedFromRedis2DB();likedService.transLikedCountFromRedis2DB();}
}

在定时任务中直接调用 LikedService 封装的方法完成数据同步。


以上就是点赞功能的设计与实现,不足之处还请各位大佬多多指教。

如有更好的实现方案欢迎在评论区交流…


代码出自开源项目 CodeRiver,致力于打造全平台型全栈精品开源项目。

coderiver 中文名 河码,是一个为程序员和设计师提供项目协作的平台。无论你是前端、后端、移动端开发人员,或是设计师、产品经理,都可以在平台上发布项目,与志同道合的小伙伴一起协作完成项目。

coderiver河码 类似程序员客栈,但主要目的是方便各细分领域人才之间技术交流,共同成长,多人协作完成项目。暂不涉及金钱交易。

计划做成包含 pc端(Vue、React)、移动H5(Vue、React)、ReactNative混合开发、Android原生、微信小程序、java后端的全平台型全栈项目,欢迎关注。

项目地址:github.com/cachecats/c…

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

WordPress回收站自动清空时间?如何关闭回收站或设置自动清理天数?

我们在WordPress后台的文章、页面、评论等页面都可以看到有回收站&#xff0c;意思就是我们不能直接删除某篇文章、页面、评论&#xff0c;而是需要现将它们移至回收站&#xff0c;然后再到回收站永久删除&#xff0c;或等回收站自动清理。 如上图所示&#xff0c;WordPress 6.…

javacv和opencv对图文视频编辑-裸眼3D图片制作

通过斗鸡眼&#xff0c;将左右两张相似的图片叠加到一起看&#xff0c;就会有3D效果。 3D图片&#xff0c;3D眼镜&#xff0c;3D视频等原理类似&#xff0c;都是通过两眼视觉差引起脑补产生3D效果。 图片&#xff1a; 图片来源&#xff1a; 一些我拍摄的真*裸眼3D照片 - 哔哩…

Java21 + SpringBoot3集成Spring Data JPA

Java21 SpringBoot3集成Spring Data JPA 文章目录 Java21 SpringBoot3集成Spring Data JPA前言相关技术简介ORM&#xff08;Object-Relational Mapping&#xff0c;对象-关系映射&#xff09;JPA&#xff08;Java Persistence API&#xff0c;Java持久层API&#xff09;Hiber…

新版K8s:v1.28拉取Harbor仓库镜像以及本地镜像(docker弃用改用containerd,纯纯踩坑)

目录 一、项目概述二、环境三、项目样式Harborkuboard运行样式 四、核心点Harbor安装config.toml文件修改(containerd)ctr、nerdctl相关命令kuboard工作负载 五、总结 一、项目概述 使用Kuboard作为k8s集群的管理平台&#xff0c;Harbor作为镜像仓库&#xff0c;拉取Harbor镜像…

【kafka】记录用-----------1

主题&#xff08;topic&#xff09;&#xff1a;消息的第一次分类 根据人为的划分条件将消息分成不同的主题 主题的划分是人为的根据不同的任务情景去划分 比如&#xff0c;我们有两个主题&#xff0c;一个是"订单"&#xff0c;另一个是"库存"。每个主题代…

24号资源——程序:电力系统程序集合已提供下载资源

24号资源&#xff1a;程序集合包含17个程序&#xff08;经典电力系统经济调度程序&#xff1b;2解决带储&#xff1b;3智能微电网PSO优化算法&#xff1b;微电网调度等等见资源描述&#xff09;资源-CSDN文库https://download.csdn.net/download/LIANG674027206/88752141&#…

ruoyi-cloud—若依微服务打包部署

1. 前端端口修改 2. 后端端口修改 &#xff08;1&#xff09;修改ruoyi-gateway服务中的bootstrap.yml的port端口 &#xff08;2&#xff09;修改ruoyi-ui中的vue.confing.js的target中的端口 3. 后端部署 (1) 在本地电脑上代码界面上打包后端 在ruoyi项目的bin目录下执行pa…

迭代器模式介绍

目录 一、迭代器模式介绍 1.1 迭代器模式定义 1.2 迭代器模式原理 1.2.1 迭代器模式类图 1.2.2 模式角色说明 1.2.3 示例代码 二、迭代模式的应用 2.1 需求说明 2.2 需求实现 2.2.1 抽象迭代类 2.2.2 抽象集合类 2.2.3 主题类 2.2.4 具体迭代类 2.2.5 具体集合类 …

【动态规划】【数学】【C++算法】18赛车

作者推荐 视频算法专题 本文涉及知识点 动态规划 数学 LeetCode818赛车 你的赛车可以从位置 0 开始&#xff0c;并且速度为 1 &#xff0c;在一条无限长的数轴上行驶。赛车也可以向负方向行驶。赛车可以按照由加速指令 ‘A’ 和倒车指令 ‘R’ 组成的指令序列自动行驶。 当…

软件工程应用题汇总

绘制数据流图(L0/L1/L2) DFD/L0&#xff08;基本系统模型&#xff09; 只包含源点终点和一个处理(XXX系统) DFD/L1&#xff08;功能级数据流图&#xff09;在L0基础上进一步划分处理(XXX系统) 个人理解 DFD/L2&#xff08;在L1基础上进一步分解后的数据流图&#xff09; 数据…

flex布局(3)

九、骰子 *{margin:0;padding: 0;box-sizing: border-box; } .flex{display: flex;flex-flow: row wrap;justify-content: space-between;align-items: center;align-content: space-between;padding:20px; } .touzi{width: 120px;height: 120px;background-color: aliceblue;…

canvas绘制美队盾牌

查看专栏目录 canvas示例教程100专栏&#xff0c;提供canvas的基础知识&#xff0c;高级动画&#xff0c;相关应用扩展等信息。canvas作为html的一部分&#xff0c;是图像图标地图可视化的一个重要的基础&#xff0c;学好了canvas&#xff0c;在其他的一些应用上将会起到非常重…

陶瓷碗口缺口检测-图像分割

图像分割 由于对碗口进行缺口检测&#xff0c;因此只需要碗口的边界信息。得到陶瓷碗区域填充后的图像&#xff0c;对图像进行边缘检测。这是属于图像分割中的内容&#xff0c;在图像的边缘中&#xff0c;可以利用导数算子对数字图像求差分&#xff0c;将边缘提取出来。 本案…

React 基于Ant Degisn 实现table表格列表拖拽排序

效果图&#xff1a; 代码&#xff1a; myRow.js import { MenuOutlined } from ant-design/icons; import { DndContext } from dnd-kit/core; import { restrictToVerticalAxis } from dnd-kit/modifiers; import {arrayMove,SortableContext,useSortable,verticalListSorti…

探案录 | 人大金仓一个底座+多场景应用

近日&#xff0c;金仓大世界发布了《2023城市数字经济发展报告》&#xff0c;福尔摩斯•K从报告中抓住了三大重点&#xff1a;第一&#xff0c;数字经济规模超过50万亿元&#xff0c;占GDP比重提升至41.5%&#xff1b;第二&#xff0c;数字经济与实体经济融合愈发紧密&#xff…

阿赵UE学习笔记——10、Blender材质和绘制网格体

阿赵UE学习笔记目录   大家好&#xff0c;我是阿赵。   之前介绍了虚幻引擎的材质和材质实例。这次来介绍一个比较有趣的内置的Blender材质。   在用Unity的时候&#xff0c;我做过一个多通道混合地表贴图的效果&#xff0c;而要做过一个刷顶点颜色混合地表和水面的效果。…

【深度学习目标检测】十六、基于深度学习的麦穗头系统-含GUI和源码(python,yolov8)

全球麦穗检测是植物表型分析领域的一个挑战&#xff0c;主要目标是检测图像中的小麦麦穗。这种检测在农业领域具有重要意义&#xff0c;可以帮助农民评估作物的健康状况和成熟度。然而&#xff0c;由于小麦麦穗在视觉上具有挑战性&#xff0c;准确检测它们是一项艰巨的任务。 全…

记redis5.x在windows上搭建集群(六主六从)

六个运行端口 127.0.0.1:6379 127.0.0.1:6380 127.0.0.1:6381 127.0.0.1:6382 127.0.0.1:6383 127.0.0.1:6384 1、安装redis,文章太多不多BB 2、复制六份redis文件夹出来改名 3、修改每一份的配置文件 redis.windows.conf 修改为以下格式&#xff1a; #运行端口 port…

CAN记录仪在矿卡中的应用

CAN数据记录仪在矿卡中主要用于记录和监控车辆的运行数据&#xff0c;以保障安全和提高运营效率。那么就需要记录整车数据来进行车辆诊断分析&#xff0c;查找问题解决问题。 CAN数据记录仪可以记录矿卡的各种运行参数&#xff0c;如发动机转速、车速、制动状态、转向状态、油…

《WebKit 技术内幕》之二: HTML 网页和结构

第二章 HTML 网页和结构 HTML网页是利用HTML语言编写的文档&#xff0c;HTML是半结构化的数据表现方式&#xff0c;它的结构特征可以归纳为&#xff1a;树状结构、层次结构和框结构。 1.网页构成 1.1 基本元素和树状结构 HTML网页使用HTML语言撰写的文档&#xff0c;发展到今…