diff --git a/Springboot_01Demo.iml b/XubxBlog.iml
similarity index 61%
rename from Springboot_01Demo.iml
rename to XubxBlog.iml
index 93f251c..88093bd 100644
--- a/Springboot_01Demo.iml
+++ b/XubxBlog.iml
@@ -2,7 +2,7 @@
-
+
\ No newline at end of file
diff --git a/src/main/java/com/xubx/springboot_01demo/Springboot01DemoApplication.java b/src/main/java/com/xubx/springboot_01demo/Springboot01DemoApplication.java
index 762c782..9efd06a 100644
--- a/src/main/java/com/xubx/springboot_01demo/Springboot01DemoApplication.java
+++ b/src/main/java/com/xubx/springboot_01demo/Springboot01DemoApplication.java
@@ -4,6 +4,7 @@ import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cache.annotation.EnableCaching;
+import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.annotation.EnableScheduling;
import springfox.documentation.swagger2.annotations.EnableSwagger2;
@@ -12,6 +13,7 @@ import springfox.documentation.swagger2.annotations.EnableSwagger2;
@MapperScan("com.xubx.springboot_01demo.mapper")
@EnableScheduling //开启定时任务
@EnableSwagger2 //开启Swagger2
+@EnableAsync // 开启异步任务
public class Springboot01DemoApplication {
public static void main(String[] args) {
diff --git a/src/main/java/com/xubx/springboot_01demo/Task/DataSync.java b/src/main/java/com/xubx/springboot_01demo/Task/DataSync.java
index add3617..c2913ac 100644
--- a/src/main/java/com/xubx/springboot_01demo/Task/DataSync.java
+++ b/src/main/java/com/xubx/springboot_01demo/Task/DataSync.java
@@ -6,40 +6,47 @@ import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.xubx.springboot_01demo.configuration.constantConfiguration;
+import com.xubx.springboot_01demo.entity.pojo.TopK;
+import com.xubx.springboot_01demo.entity.vo.HotBlog;
import com.xubx.springboot_01demo.mapper.BlogLikeMapper;
import com.xubx.springboot_01demo.mapper.BlogsMapper;
import com.xubx.springboot_01demo.mapper.UserFavoriteBlogMapper;
-import com.xubx.springboot_01demo.pojo.BlogLike;
-import com.xubx.springboot_01demo.pojo.Blogs;
-import com.xubx.springboot_01demo.pojo.UserFavoriteBlog;
+import com.xubx.springboot_01demo.entity.pojo.BlogLike;
+import com.xubx.springboot_01demo.entity.pojo.Blogs;
+import com.xubx.springboot_01demo.entity.pojo.UserFavoriteBlog;
+import com.xubx.springboot_01demo.service.BlogService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.data.redis.core.RedisCallback;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
+import org.springframework.util.ObjectUtils;
import javax.annotation.Resource;
+import java.nio.charset.StandardCharsets;
import java.sql.Timestamp;
import java.time.LocalDateTime;
-import java.util.ArrayList;
-import java.util.Date;
-import java.util.List;
-import java.util.Set;
+import java.util.*;
import java.util.concurrent.TimeUnit;
+import java.util.stream.Collectors;
@Component
@Slf4j
public class DataSync {
@Resource
- RedisTemplate redisTemplate;
+ private RedisTemplate redisTemplate;
@Resource
- BlogsMapper blogsMapper;
+ private BlogsMapper blogsMapper;
@Resource
- BlogLikeMapper blogLikeMapper;
+ private BlogLikeMapper blogLikeMapper;
@Resource
- UserFavoriteBlogMapper userFavoriteBlogMapper;
+ private UserFavoriteBlogMapper userFavoriteBlogMapper;
@Autowired
- ObjectMapper objectMapper;
+ private BlogService blogService;
+ @Autowired
+ private ObjectMapper objectMapper;
+
/**
* 博客浏览数的数据同步 ,每小时执行一次
@@ -97,6 +104,7 @@ public class DataSync {
try {
Set likeCountKeys = redisTemplate.keys(constantConfiguration.BLOG_LIKE_COUNT + "*");
if (likeCountKeys != null) {
+ likeCountKeys = likeCountKeys.stream().filter(key -> !key.endsWith(":expired")).collect(Collectors.toSet());
for (String key : likeCountKeys) {
String blogIdStr = key.split(":")[2];
String likeCountExpireKey = constantConfiguration.BLOG_LIKE_COUNT + blogIdStr + ":expired";
@@ -139,6 +147,7 @@ public class DataSync {
try {
Set likeUserKeys = redisTemplate.keys(constantConfiguration.BLOG_LIKE_HASH + "*");
if (likeUserKeys != null) {
+ likeUserKeys = likeUserKeys.stream().filter(key -> !key.endsWith(":expired")).collect(Collectors.toSet());
for (String key : likeUserKeys) {
String blogIdStr = key.split(":")[2];
String likeUserExpireKey = constantConfiguration.BLOG_LIKE_HASH + blogIdStr + ":expired";
@@ -160,10 +169,10 @@ public class DataSync {
if (blogLike == null) {
BlogLike newBlogLike = new BlogLike();
- newBlogLike.setBlogId(Integer.valueOf(blogIdStr));
- newBlogLike.setUserId(Integer.valueOf(userId));
+ newBlogLike.setBlogId(Long.valueOf(blogIdStr));
+ newBlogLike.setUserId(Long.valueOf(userId));
newBlogLike.setCreatedTime(Timestamp.valueOf(LocalDateTime.now()));
- blogLikeMapper.insert(newBlogLike);
+ blogLikeMapper.addData(newBlogLike);
}
} else if ("false".equals(likeStatus)) {
// 用户取消点赞 则删除记录
@@ -190,7 +199,7 @@ public class DataSync {
/**
* 博客收藏数的数据同步定时任务
*/
- @Scheduled(cron = "0 * * * * ?") //TODO 每小时执行一次
+ @Scheduled(cron = "0 * * * * ?") //TODO 每小时执行一次,需要隔开,不能同一时间全部执行
public void collectCountSync() {
log.info("博客收藏数的数据同步定时任务执行");
String lockKey = constantConfiguration.BLOG_COLLECT_COUNT_SYNC_LOCK;
@@ -200,6 +209,7 @@ public class DataSync {
try {
Set collectCountKeys = redisTemplate.keys(constantConfiguration.BLOG_COLLECT_COUNT + "*");
if (collectCountKeys != null) {
+ collectCountKeys = collectCountKeys.stream().filter(key -> !key.endsWith(":expired")).collect(Collectors.toSet());
for (String key : collectCountKeys) {
String blogIdStr = key.split(":")[2];
String collectCountExpireKey = constantConfiguration.BLOG_COLLECT_COUNT + blogIdStr + ":expired";
@@ -245,6 +255,7 @@ public class DataSync {
// TODO 应该是同时获取到了逻辑过期标志的Key,需要进行去除
Set collectUserKeys = redisTemplate.keys(constantConfiguration.BLOG_COLLECT_HASH + "*");
if (collectUserKeys != null) {
+ collectUserKeys = collectUserKeys.stream().filter(key -> !key.contains(":expired")).collect(Collectors.toSet());
for (String key : collectUserKeys) {
String favoriteIdStr = key.split(":")[2];
String collectUserExpireKey = constantConfiguration.BLOG_COLLECT_HASH + favoriteIdStr + ":expired";
@@ -291,4 +302,77 @@ public class DataSync {
log.warn("无法获取分布式锁,跳过此次收藏用户Hash集合的数据同步任务");
}
}
+
+ /**
+ * 热度排行榜,全表分片扫描
+ */
+ @Scheduled(cron = "0 * * * * ?") //TODO 每小时执行一次
+ public void hotRank() {
+ log.info("热度排行榜刷新定时任务执行");
+
+ final TopK topK = new TopK(10);
+ long limit = 1000;
+ long id = 0;
+
+ List blogs = blogService.list(new LambdaQueryWrapper()
+ .select(Blogs::getId, Blogs::getTitle, Blogs::getLikeCount, Blogs::getViewCount, Blogs::getCollectCount, Blogs::getCommentCount,
+ Blogs::getShareCount, Blogs::getCreatedTime).gt(Blogs::getId, id)
+ .eq(Blogs::getStatus, "公开")
+ .last("limit " + limit));
+
+ while (!ObjectUtils.isEmpty(blogs)) {
+ for (Blogs blog : blogs) {
+ Long likeCount = blog.getLikeCount();
+ Double viewCount = blog.getViewCount() * 0.8;
+ Double collectCount = blog.getCollectCount() * 1.5;
+ Long commentCount = blog.getCommentCount();
+ Long shareCount = blog.getShareCount();
+ final Date date = new Date();
+ long decayTime = date.getTime() - blog.getCreatedTime().getTime();
+ // 防止出现热度完全相同的情况,加入0.1-1.0之间的随机小数
+ final double v = Math.random() * 0.9 + 0.1;
+ final double hot = hot(likeCount + viewCount + collectCount + commentCount + shareCount + v, TimeUnit.MILLISECONDS.toDays(decayTime));
+ final HotBlog hotBlog = new HotBlog(blog.getId(), hot, blog.getTitle());
+
+ // add过程中,已存入热度最大的十个博客
+ topK.add(hotBlog);
+ }
+ id = blogs.get(blogs.size() - 1).getId();
+ blogs = blogService.list(new LambdaQueryWrapper().gt(Blogs::getId, id).eq(Blogs::getStatus, "公开").last("limit " + limit));
+ }
+ // 通过redis管道进行数据更新
+ final byte[] key = constantConfiguration.HOT_RANK.getBytes();
+ final List hotBlogs = topK.get();// 已从大到小排序
+ final Double minHot = hotBlogs.get(hotBlogs.size() - 1).getHot();
+ redisTemplate.executePipelined((RedisCallback