点赞,收藏,浏览模块
This commit is contained in:
parent
1510401ccc
commit
300f326281
|
@ -7,13 +7,20 @@
|
||||||
<sourceOutputDir name="target/generated-sources/annotations" />
|
<sourceOutputDir name="target/generated-sources/annotations" />
|
||||||
<sourceTestOutputDir name="target/generated-test-sources/test-annotations" />
|
<sourceTestOutputDir name="target/generated-test-sources/test-annotations" />
|
||||||
<outputRelativeToContentRoot value="true" />
|
<outputRelativeToContentRoot value="true" />
|
||||||
<module name="Springboot_01Demo" />
|
<module name="XubxBlog" />
|
||||||
</profile>
|
</profile>
|
||||||
</annotationProcessing>
|
</annotationProcessing>
|
||||||
|
<bytecodeTargetLevel>
|
||||||
|
<module name="blog-xubx-parent" target="1.5" />
|
||||||
|
<module name="xubx-blog-api" target="1.5" />
|
||||||
|
<module name="xubx-blog-parent" target="1.6" />
|
||||||
|
</bytecodeTargetLevel>
|
||||||
</component>
|
</component>
|
||||||
<component name="JavacSettings">
|
<component name="JavacSettings">
|
||||||
<option name="ADDITIONAL_OPTIONS_OVERRIDE">
|
<option name="ADDITIONAL_OPTIONS_OVERRIDE">
|
||||||
<module name="Springboot_01Demo" options="-parameters" />
|
<module name="XubxBlog" options="-parameters" />
|
||||||
|
<module name="xubx-blog-api" options="" />
|
||||||
|
<module name="xubx-blog-parent" options="" />
|
||||||
</option>
|
</option>
|
||||||
</component>
|
</component>
|
||||||
</project>
|
</project>
|
5
pom.xml
5
pom.xml
|
@ -91,6 +91,11 @@
|
||||||
<artifactId>fastjson</artifactId>
|
<artifactId>fastjson</artifactId>
|
||||||
<version>1.2.83</version>
|
<version>1.2.83</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.fasterxml.jackson.core</groupId>
|
||||||
|
<artifactId>jackson-databind</artifactId>
|
||||||
|
<version>2.12.5</version>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>io.springfox</groupId>
|
<groupId>io.springfox</groupId>
|
||||||
<artifactId>springfox-swagger2</artifactId>
|
<artifactId>springfox-swagger2</artifactId>
|
||||||
|
|
|
@ -0,0 +1,294 @@
|
||||||
|
package com.xubx.springboot_01demo.Task;
|
||||||
|
|
||||||
|
import com.alibaba.fastjson.JSON;
|
||||||
|
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
||||||
|
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.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 lombok.extern.slf4j.Slf4j;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.data.redis.core.RedisTemplate;
|
||||||
|
import org.springframework.scheduling.annotation.Scheduled;
|
||||||
|
import org.springframework.stereotype.Component;
|
||||||
|
|
||||||
|
import javax.annotation.Resource;
|
||||||
|
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.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
@Component
|
||||||
|
@Slf4j
|
||||||
|
public class DataSync {
|
||||||
|
@Resource
|
||||||
|
RedisTemplate redisTemplate;
|
||||||
|
@Resource
|
||||||
|
BlogsMapper blogsMapper;
|
||||||
|
@Resource
|
||||||
|
BlogLikeMapper blogLikeMapper;
|
||||||
|
@Resource
|
||||||
|
UserFavoriteBlogMapper userFavoriteBlogMapper;
|
||||||
|
@Autowired
|
||||||
|
ObjectMapper objectMapper;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 博客浏览数的数据同步 ,每小时执行一次
|
||||||
|
*/
|
||||||
|
// @Scheduled(cron = "0 0 * * * ?")
|
||||||
|
@Scheduled(cron = "0 * * * * ?")
|
||||||
|
public void viewCountSync() {
|
||||||
|
log.info("博客浏览数的数据同步定时任务执行");
|
||||||
|
|
||||||
|
String lockKey = constantConfiguration.BLOG_VIEW_COUNT_SYNC_LOCK;
|
||||||
|
Boolean isLock = Boolean.TRUE.equals(redisTemplate.opsForValue().setIfAbsent(lockKey, "1", 60, TimeUnit.SECONDS));
|
||||||
|
if (Boolean.TRUE.equals(isLock)) {
|
||||||
|
try {
|
||||||
|
Set<String> viewCountKeys = redisTemplate.keys(constantConfiguration.BLOG_VIEW_COUNT + "*");
|
||||||
|
if (viewCountKeys != null) {
|
||||||
|
for (String key : viewCountKeys) {
|
||||||
|
String blogIdStr = key.split(":")[2];
|
||||||
|
String expirekey = constantConfiguration.BLOG_VIEW_COUNT + blogIdStr + ":expired";
|
||||||
|
|
||||||
|
Boolean isExpired = Boolean.TRUE.equals(redisTemplate.hasKey(expirekey))
|
||||||
|
&& Boolean.parseBoolean((String) redisTemplate.opsForValue().get(expirekey));
|
||||||
|
// 如果标签存在 则进行数据的更新,若不存在则表示数据已经同步 或者数据还没被更新
|
||||||
|
if (Boolean.TRUE.equals(isExpired)) {
|
||||||
|
String viewCount = (String) redisTemplate.opsForValue().get(key);
|
||||||
|
if (viewCount != null) {
|
||||||
|
//更新数据
|
||||||
|
LambdaUpdateWrapper<Blogs> updateWrapper = new LambdaUpdateWrapper<>();
|
||||||
|
updateWrapper.eq(Blogs::getId, blogIdStr)
|
||||||
|
.set(Blogs::getViewCount, viewCount);
|
||||||
|
blogsMapper.update(null, updateWrapper);
|
||||||
|
// 删除逻辑过期标志
|
||||||
|
redisTemplate.delete(expirekey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
redisTemplate.delete(lockKey);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.warn("无法获取分布式锁,博客浏览数的数据同步任务无法执行");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 博客点赞数的数据同步定时任务执行
|
||||||
|
*/
|
||||||
|
@Scheduled(cron = "0 * * * * ?")
|
||||||
|
public void likeCountSync() {
|
||||||
|
log.info("博客点赞数的数据同步定时任务执行");
|
||||||
|
String lockKey = constantConfiguration.BLOG_LIKE_COUNT_SYNC_LOCK;
|
||||||
|
Boolean isLock = Boolean.TRUE.equals(redisTemplate.opsForValue().setIfAbsent(lockKey, "1", 60, TimeUnit.SECONDS));
|
||||||
|
|
||||||
|
if (Boolean.TRUE.equals(isLock)) {
|
||||||
|
try {
|
||||||
|
Set<String> likeCountKeys = redisTemplate.keys(constantConfiguration.BLOG_LIKE_COUNT + "*");
|
||||||
|
if (likeCountKeys != null) {
|
||||||
|
for (String key : likeCountKeys) {
|
||||||
|
String blogIdStr = key.split(":")[2];
|
||||||
|
String likeCountExpireKey = constantConfiguration.BLOG_LIKE_COUNT + blogIdStr + ":expired";
|
||||||
|
|
||||||
|
Boolean countExpireFlag = Boolean.TRUE.equals(redisTemplate.hasKey(likeCountExpireKey))
|
||||||
|
&& Boolean.parseBoolean((String) redisTemplate.opsForValue().get(likeCountExpireKey));
|
||||||
|
|
||||||
|
if (Boolean.TRUE.equals(countExpireFlag)) {
|
||||||
|
String likeCount = (String) redisTemplate.opsForValue().get(key);
|
||||||
|
if (likeCount != null) {
|
||||||
|
// 更新点赞次数到数据库
|
||||||
|
LambdaUpdateWrapper<Blogs> updateWrapper = new LambdaUpdateWrapper<>();
|
||||||
|
updateWrapper.eq(Blogs::getId, blogIdStr).set(Blogs::getLikeCount, Integer.parseInt(likeCount));
|
||||||
|
blogsMapper.update(null, updateWrapper);
|
||||||
|
|
||||||
|
// 删除逻辑过期标志
|
||||||
|
redisTemplate.delete(likeCountExpireKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
redisTemplate.delete(lockKey);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.warn("无法获取分布式锁,博客点赞数的数据同步任务无法执行");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 博客点赞用户Hash集合的增量数据同步定时任务执行
|
||||||
|
*/
|
||||||
|
@Scheduled(cron = "0 * * * * ?") //TODO 每小时执行一次
|
||||||
|
public void likeUserSync() {
|
||||||
|
log.info("博客点赞用户Hash集合的增量数据同步定时任务执行");
|
||||||
|
String lockKey = constantConfiguration.BLOG_LIKE_USERHASH_SYNC_LOCK;
|
||||||
|
Boolean isLock = Boolean.TRUE.equals(redisTemplate.opsForValue().setIfAbsent(lockKey, "1", 60, TimeUnit.SECONDS));
|
||||||
|
|
||||||
|
if (Boolean.TRUE.equals(isLock)) {
|
||||||
|
try {
|
||||||
|
Set<String> likeUserKeys = redisTemplate.keys(constantConfiguration.BLOG_LIKE_HASH + "*");
|
||||||
|
if (likeUserKeys != null) {
|
||||||
|
for (String key : likeUserKeys) {
|
||||||
|
String blogIdStr = key.split(":")[2];
|
||||||
|
String likeUserExpireKey = constantConfiguration.BLOG_LIKE_HASH + blogIdStr + ":expired";
|
||||||
|
|
||||||
|
Boolean userHashExpireFlag = Boolean.TRUE.equals(redisTemplate.hasKey(likeUserExpireKey))
|
||||||
|
&& Boolean.parseBoolean((String) redisTemplate.opsForValue().get(likeUserExpireKey));
|
||||||
|
|
||||||
|
if (Boolean.TRUE.equals(userHashExpireFlag)) {
|
||||||
|
// 获取博客点赞用户集合中的所有用户ID
|
||||||
|
Set<String> userIds = redisTemplate.opsForHash().keys(constantConfiguration.BLOG_LIKE_HASH + blogIdStr);
|
||||||
|
if (userIds != null) {
|
||||||
|
for (String userId : userIds) {
|
||||||
|
String likeStatus = (String) redisTemplate.opsForHash().get(constantConfiguration.BLOG_LIKE_HASH + blogIdStr, userId);
|
||||||
|
if ("true".equals(likeStatus)) {
|
||||||
|
// 插入数据
|
||||||
|
LambdaQueryWrapper<BlogLike> queryWrapper = new LambdaQueryWrapper<>();
|
||||||
|
queryWrapper.eq(BlogLike::getBlogId, blogIdStr).eq(BlogLike::getUserId, userId);
|
||||||
|
BlogLike blogLike = blogLikeMapper.selectOne(queryWrapper);
|
||||||
|
|
||||||
|
if (blogLike == null) {
|
||||||
|
BlogLike newBlogLike = new BlogLike();
|
||||||
|
newBlogLike.setBlogId(Integer.valueOf(blogIdStr));
|
||||||
|
newBlogLike.setUserId(Integer.valueOf(userId));
|
||||||
|
newBlogLike.setCreatedTime(Timestamp.valueOf(LocalDateTime.now()));
|
||||||
|
blogLikeMapper.insert(newBlogLike);
|
||||||
|
}
|
||||||
|
} else if ("false".equals(likeStatus)) {
|
||||||
|
// 用户取消点赞 则删除记录
|
||||||
|
LambdaQueryWrapper<BlogLike> queryWrapper = new LambdaQueryWrapper<>();
|
||||||
|
queryWrapper.eq(BlogLike::getBlogId, blogIdStr).eq(BlogLike::getUserId, userId);
|
||||||
|
blogLikeMapper.delete(queryWrapper);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 删除逻辑过期标志和哈希数据
|
||||||
|
redisTemplate.delete(likeUserExpireKey);
|
||||||
|
redisTemplate.delete(constantConfiguration.BLOG_LIKE_HASH + blogIdStr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
redisTemplate.delete(lockKey);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.warn("无法获取分布式锁,博客点赞用户Hash集合的增量数据同步任务无法执行");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 博客收藏数的数据同步定时任务
|
||||||
|
*/
|
||||||
|
@Scheduled(cron = "0 * * * * ?") //TODO 每小时执行一次
|
||||||
|
public void collectCountSync() {
|
||||||
|
log.info("博客收藏数的数据同步定时任务执行");
|
||||||
|
String lockKey = constantConfiguration.BLOG_COLLECT_COUNT_SYNC_LOCK;
|
||||||
|
Boolean lockAcquired = redisTemplate.opsForValue().setIfAbsent(lockKey, "LOCKED", 30, TimeUnit.MINUTES);
|
||||||
|
|
||||||
|
if (Boolean.TRUE.equals(lockAcquired)) {
|
||||||
|
try {
|
||||||
|
Set<String> collectCountKeys = redisTemplate.keys(constantConfiguration.BLOG_COLLECT_COUNT + "*");
|
||||||
|
if (collectCountKeys != null) {
|
||||||
|
for (String key : collectCountKeys) {
|
||||||
|
String blogIdStr = key.split(":")[2];
|
||||||
|
String collectCountExpireKey = constantConfiguration.BLOG_COLLECT_COUNT + blogIdStr + ":expired";
|
||||||
|
|
||||||
|
Boolean countExpireFlag = Boolean.TRUE.equals(redisTemplate.hasKey(collectCountExpireKey))
|
||||||
|
&& Boolean.parseBoolean((String) redisTemplate.opsForValue().get(collectCountExpireKey));
|
||||||
|
|
||||||
|
if (Boolean.TRUE.equals(countExpireFlag)) {
|
||||||
|
String collectCount = (String) redisTemplate.opsForValue().get(key);
|
||||||
|
if (collectCount != null) {
|
||||||
|
// 更新收藏次数到数据库
|
||||||
|
LambdaUpdateWrapper<Blogs> updateWrapper = new LambdaUpdateWrapper<>();
|
||||||
|
updateWrapper.eq(Blogs::getId, blogIdStr).set(Blogs::getCollectCount, Integer.parseInt(collectCount));
|
||||||
|
blogsMapper.update(null, updateWrapper);
|
||||||
|
|
||||||
|
// 删除逻辑过期标志
|
||||||
|
redisTemplate.delete(collectCountExpireKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
redisTemplate.delete(lockKey);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.warn("无法获取分布式锁,跳过此次收藏数同步任务");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 博客收藏增量数据同步
|
||||||
|
*
|
||||||
|
* @throws JsonProcessingException
|
||||||
|
*/
|
||||||
|
@Scheduled(cron = "0 * * * * ?") //TODO 每小时执行一次
|
||||||
|
public void collectUserSync() {
|
||||||
|
log.info("博客收藏用户Hash集合的数据同步定时任务执行");
|
||||||
|
String lockKey = constantConfiguration.BLOG_COLLECT_USERHASH_SYNC_LOCK;
|
||||||
|
Boolean lockAcquired = redisTemplate.opsForValue().setIfAbsent(lockKey, "LOCKED", 30, TimeUnit.MINUTES);
|
||||||
|
|
||||||
|
if (Boolean.TRUE.equals(lockAcquired)) {
|
||||||
|
try {
|
||||||
|
// TODO 应该是同时获取到了逻辑过期标志的Key,需要进行去除
|
||||||
|
Set<String> collectUserKeys = redisTemplate.keys(constantConfiguration.BLOG_COLLECT_HASH + "*");
|
||||||
|
if (collectUserKeys != null) {
|
||||||
|
for (String key : collectUserKeys) {
|
||||||
|
String favoriteIdStr = key.split(":")[2];
|
||||||
|
String collectUserExpireKey = constantConfiguration.BLOG_COLLECT_HASH + favoriteIdStr + ":expired";
|
||||||
|
|
||||||
|
Boolean userExpireFlag = Boolean.TRUE.equals(redisTemplate.hasKey(collectUserExpireKey))
|
||||||
|
&& Boolean.parseBoolean((String) redisTemplate.opsForValue().get(collectUserExpireKey));
|
||||||
|
if (Boolean.TRUE.equals(userExpireFlag)) {
|
||||||
|
// 获取博客收藏用户集合中的所有博客
|
||||||
|
Set<String> blogSet = redisTemplate.opsForHash().keys(constantConfiguration.BLOG_COLLECT_HASH + favoriteIdStr);
|
||||||
|
if (blogSet != null) {
|
||||||
|
for (String blogIdStr : blogSet) {
|
||||||
|
// 根据用户的收藏状态,更新数据
|
||||||
|
String collectStatus = (String) redisTemplate.opsForHash().get(constantConfiguration.BLOG_COLLECT_HASH + favoriteIdStr, blogIdStr);
|
||||||
|
if ("true".equals(collectStatus)) {
|
||||||
|
// 如果数据库中没有该数据则插入数据
|
||||||
|
LambdaQueryWrapper<UserFavoriteBlog> queryWrapper = new LambdaQueryWrapper<>();
|
||||||
|
queryWrapper.eq(UserFavoriteBlog::getBlogId, blogIdStr).eq(UserFavoriteBlog::getFavoriteId, favoriteIdStr);
|
||||||
|
UserFavoriteBlog userFavoriteBlog = userFavoriteBlogMapper.selectOne(queryWrapper);
|
||||||
|
if (userFavoriteBlog == null) {
|
||||||
|
userFavoriteBlog = new UserFavoriteBlog();
|
||||||
|
userFavoriteBlog.setBlogId(Integer.valueOf(blogIdStr));
|
||||||
|
userFavoriteBlog.setFavoriteId(Integer.valueOf(favoriteIdStr));
|
||||||
|
userFavoriteBlogMapper.insert(userFavoriteBlog);
|
||||||
|
}
|
||||||
|
} else if ("false".equals(collectStatus)) {
|
||||||
|
// 用户取消收藏 删除数据
|
||||||
|
LambdaQueryWrapper<UserFavoriteBlog> queryWrapper = new LambdaQueryWrapper<>();
|
||||||
|
queryWrapper.eq(UserFavoriteBlog::getBlogId, blogIdStr).eq(UserFavoriteBlog::getFavoriteId, favoriteIdStr);
|
||||||
|
userFavoriteBlogMapper.delete(queryWrapper);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 删除逻辑过期标志和Hash集合
|
||||||
|
redisTemplate.delete(collectUserExpireKey);
|
||||||
|
redisTemplate.delete(constantConfiguration.BLOG_COLLECT_HASH + favoriteIdStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
redisTemplate.delete(lockKey);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.warn("无法获取分布式锁,跳过此次收藏用户Hash集合的数据同步任务");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,27 @@
|
||||||
|
package com.xubx.springboot_01demo.configuration;
|
||||||
|
|
||||||
|
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.serializer.StringRedisSerializer;
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
public class RedisConfig {
|
||||||
|
@Bean
|
||||||
|
public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory connectionFactory) {
|
||||||
|
RedisTemplate<String, Object> template = new RedisTemplate<>();
|
||||||
|
template.setConnectionFactory(connectionFactory);
|
||||||
|
|
||||||
|
// 设置 Key 和 Value 的序列化方式为 String
|
||||||
|
template.setKeySerializer(new StringRedisSerializer());
|
||||||
|
template.setValueSerializer(new StringRedisSerializer());
|
||||||
|
|
||||||
|
// 设置 Hash Key 和 Hash Value 的序列化方式为 String
|
||||||
|
template.setHashKeySerializer(new StringRedisSerializer());
|
||||||
|
template.setHashValueSerializer(new StringRedisSerializer());
|
||||||
|
|
||||||
|
template.afterPropertiesSet();
|
||||||
|
return template;
|
||||||
|
}
|
||||||
|
}
|
|
@ -13,4 +13,47 @@ public class constantConfiguration {
|
||||||
public static final int Offline = 0;
|
public static final int Offline = 0;
|
||||||
//用户在线
|
//用户在线
|
||||||
public static final int Online = 1;
|
public static final int Online = 1;
|
||||||
|
// 博客点赞数
|
||||||
|
public static final String BLOG_LIKE_COUNT = "blog:likeCount:";
|
||||||
|
// 用户的总被点赞数
|
||||||
|
public static final String USER_TOTAL_LIKE_COUNT = "user:totalLikeCount:";
|
||||||
|
// 博客点赞用户集合
|
||||||
|
public static final String BLOG_LIKE_SET = "blog:likeUsers:";
|
||||||
|
// 博客点赞用户哈希
|
||||||
|
public static final String BLOG_LIKE_HASH = "blog:likeHash:";
|
||||||
|
public static final String BLOG_LIKE_COUNT_SYNC_LOCK = "blog:likeCountSyncLock:";
|
||||||
|
public static final String BLOG_LIKE_USERHASH_SYNC_LOCK = "blog:likeUserHashSyncLock:";
|
||||||
|
|
||||||
|
// 博客浏览数
|
||||||
|
public static final String BLOG_VIEW_COUNT = "blog:viewCount:";
|
||||||
|
// 博客浏览用户集合
|
||||||
|
public static final String BLOG_VIEW_SET = "blog:viewUsers:";
|
||||||
|
public static final String BLOG_VIEW_COUNT_SYNC_LOCK = "blog:viewCountSyncLock:";
|
||||||
|
|
||||||
|
//用户的总访问量
|
||||||
|
public static final String USER_TOTAL_VIEW_COUNT = "user:totalViewCount:";
|
||||||
|
//用户的博客总数量
|
||||||
|
public static final String USER_TOTAL_BLOG_COUNT = "user:totalBlogCount:";
|
||||||
|
|
||||||
|
// 博客评论数
|
||||||
|
public static final String BLOG_COMMENT_COUNT = "blog:commentCount:";
|
||||||
|
// 用户的总被评论数
|
||||||
|
public static final String USER_TOTAL_COMMENT_COUNT = "user:totalCommentCount:";
|
||||||
|
|
||||||
|
// 博客收藏数
|
||||||
|
public static final String BLOG_COLLECT_COUNT = "blog:collectCount:";
|
||||||
|
// 用户的总被收藏数
|
||||||
|
public static final String USER_TOTAL_COLLECT_COUNT = "user:totalCollectCount:";
|
||||||
|
// 博客收藏用户集合
|
||||||
|
public static final String BLOG_COLLECT_SET = "blog:collectUsers:";
|
||||||
|
public static final String BLOG_COLLECT_HASH = "blog:collectHash:";
|
||||||
|
public static final String BLOG_COLLECT_LOCK = "blog:collectLock:";
|
||||||
|
public static final String BLOG_COLLECT_COUNT_SYNC_LOCK = "blog:collectCountSyncLock:";
|
||||||
|
public static final String BLOG_COLLECT_USERHASH_SYNC_LOCK = "blog:collectUserHashSyncLock:";
|
||||||
|
|
||||||
|
//用户的总被分享数
|
||||||
|
public static final String USER_TOTAL_SHARE_COUNT = "user:totalShareCount:";
|
||||||
|
|
||||||
|
// 用户的总粉丝数量
|
||||||
|
public static final String USER_TOTAL_FOLLOWER_COUNT = "user:totalFollowerCount:";
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,15 +1,17 @@
|
||||||
package com.xubx.springboot_01demo.controller;
|
package com.xubx.springboot_01demo.controller;
|
||||||
|
|
||||||
import com.alibaba.fastjson.JSONObject;
|
import com.alibaba.fastjson.JSONObject;
|
||||||
import com.xubx.springboot_01demo.dto.blog.addBlogDto;
|
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
||||||
|
import com.xubx.springboot_01demo.dto.blog.AddBlogDto;
|
||||||
|
import com.xubx.springboot_01demo.dto.blog.CollectBlogDto;
|
||||||
|
import com.xubx.springboot_01demo.dto.blog.GetUserBlogsDto;
|
||||||
|
import com.xubx.springboot_01demo.dto.user.NewFavoritesDto;
|
||||||
import com.xubx.springboot_01demo.pojo.Blogs;
|
import com.xubx.springboot_01demo.pojo.Blogs;
|
||||||
import com.xubx.springboot_01demo.service.BlogService;
|
import com.xubx.springboot_01demo.service.BlogService;
|
||||||
import com.xubx.springboot_01demo.utils.api.Result;
|
import com.xubx.springboot_01demo.vo.GetBlogDetailVo;
|
||||||
import io.swagger.annotations.Api;
|
import io.swagger.annotations.Api;
|
||||||
import io.swagger.annotations.ApiOperation;
|
import io.swagger.annotations.ApiOperation;
|
||||||
import lombok.extern.slf4j.Slf4j;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
import org.springframework.cache.annotation.CacheEvict;
|
|
||||||
import org.springframework.cache.annotation.Cacheable;
|
|
||||||
import org.springframework.http.HttpStatus;
|
import org.springframework.http.HttpStatus;
|
||||||
import org.springframework.http.ResponseEntity;
|
import org.springframework.http.ResponseEntity;
|
||||||
import org.springframework.validation.BindingResult;
|
import org.springframework.validation.BindingResult;
|
||||||
|
@ -41,20 +43,84 @@ public class BlogsController {
|
||||||
/**
|
/**
|
||||||
* 获取博客详情
|
* 获取博客详情
|
||||||
*
|
*
|
||||||
* @param id
|
* @param blogId
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
@GetMapping("/getBlogDetail")
|
@GetMapping("/getBlogDetail")
|
||||||
@ApiOperation("获取博客详情")
|
@ApiOperation("获取博客详情")
|
||||||
public ResponseEntity<Blogs> getBlogDetail(@RequestParam("blogId") int id) {
|
public ResponseEntity<?> getBlogDetail(@RequestParam("blogId") int blogId) {
|
||||||
log.info("获取博客详情,ID: {}", id);
|
log.info("获取博客详情,ID: {}", blogId);
|
||||||
Blogs blog = blogService.findByIdBlogs(id);
|
GetBlogDetailVo getBlogDetailVo = blogService.findByIdBlogs(blogId);
|
||||||
if (blog == null) {
|
|
||||||
return ResponseEntity.status(HttpStatus.NOT_FOUND).body(null);
|
if (getBlogDetailVo == null) {
|
||||||
|
return ResponseEntity.status(HttpStatus.NOT_FOUND).body("博客不存在");
|
||||||
}
|
}
|
||||||
return ResponseEntity.ok(blog);
|
return ResponseEntity.ok(getBlogDetailVo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 浏览博客,增加阅读数
|
||||||
|
*
|
||||||
|
* @param blogId
|
||||||
|
* @return {@link ResponseEntity }<{@link String }>
|
||||||
|
*/
|
||||||
|
@GetMapping("/viewBlog")
|
||||||
|
@ApiOperation("浏览博客")
|
||||||
|
public ResponseEntity<String> viewBlog(@RequestParam("blogId") int blogId) {
|
||||||
|
log.info("浏览博客,ID: {}", blogId);
|
||||||
|
|
||||||
|
try {
|
||||||
|
blogService.viewBlog(blogId);
|
||||||
|
return ResponseEntity.ok("博客浏览成功");
|
||||||
|
} catch (Exception e) {
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body("博客浏览失败");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 当前用户点赞博客
|
||||||
|
*
|
||||||
|
* @param blogId
|
||||||
|
* @return {@link ResponseEntity }<{@link String }>
|
||||||
|
*/
|
||||||
|
@GetMapping("/likeBlog")
|
||||||
|
@ApiOperation("点赞博客")
|
||||||
|
public ResponseEntity<String> likeBlog(@RequestParam("blogId") int blogId) {
|
||||||
|
log.info("点赞博客,ID: {}", blogId);
|
||||||
|
|
||||||
|
try {
|
||||||
|
blogService.likeBlog(blogId);
|
||||||
|
return ResponseEntity.ok("接口调用成功");
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("博客点赞失败", e);
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body("博客点赞失败");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 收藏博客
|
||||||
|
* @param collectBlogDto
|
||||||
|
* @return {@link ResponseEntity }<{@link String }>
|
||||||
|
*/
|
||||||
|
@PostMapping("/collectBlog")
|
||||||
|
@ApiOperation("收藏博客")
|
||||||
|
public ResponseEntity<String> collectBlog(@Valid @RequestBody CollectBlogDto collectBlogDto, BindingResult bindingResult) {
|
||||||
|
log.info("收藏博客,ID: {}", collectBlogDto);
|
||||||
|
|
||||||
|
if (bindingResult.hasErrors()) {
|
||||||
|
return ResponseEntity.status(HttpStatus.BAD_REQUEST).body(bindingResult.getFieldError().getDefaultMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
blogService.collectBlog(collectBlogDto);
|
||||||
|
return ResponseEntity.ok("接口调用成功");
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("博客收藏失败", e);
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body("博客收藏失败");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 新增博客
|
* 新增博客
|
||||||
*
|
*
|
||||||
|
@ -62,7 +128,7 @@ public class BlogsController {
|
||||||
*/
|
*/
|
||||||
@PostMapping("/addBlog")
|
@PostMapping("/addBlog")
|
||||||
@ApiOperation("新增博客")
|
@ApiOperation("新增博客")
|
||||||
public ResponseEntity<String> addBlog(@Valid @RequestBody addBlogDto blogs, BindingResult bindingResult) {
|
public ResponseEntity<String> addBlog(@Valid @RequestBody AddBlogDto blogs, BindingResult bindingResult) {
|
||||||
log.info("新增博客入参:{}", JSONObject.toJSONString(blogs));
|
log.info("新增博客入参:{}", JSONObject.toJSONString(blogs));
|
||||||
|
|
||||||
if (bindingResult.hasErrors()) {
|
if (bindingResult.hasErrors()) {
|
||||||
|
@ -101,5 +167,22 @@ public class BlogsController {
|
||||||
return ResponseEntity.ok("博客删除成功");
|
return ResponseEntity.ok("博客删除成功");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@PostMapping("/getUserBlogs")
|
||||||
|
@ApiOperation("获取用户博客")
|
||||||
|
public ResponseEntity<Page<Blogs>> getUserBlogs(@Valid @RequestBody GetUserBlogsDto getUserBlogsDto, BindingResult bindingResult) {
|
||||||
|
log.info("获取用户博客,{}", getUserBlogsDto);
|
||||||
|
|
||||||
|
if (bindingResult.hasErrors()) {
|
||||||
|
return ResponseEntity.status(HttpStatus.BAD_REQUEST).body(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
Page<Blogs> blogs = blogService.findBlogsByUserId(getUserBlogsDto);
|
||||||
|
return ResponseEntity.ok(blogs);
|
||||||
|
} catch (Exception e) {
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body(null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,7 +29,7 @@ public class MessagesController {
|
||||||
@PostMapping("/sendMessages")
|
@PostMapping("/sendMessages")
|
||||||
public ResponseEntity<Result<Void>> sendMessages(@RequestBody SendMesDto sendMesDto) {
|
public ResponseEntity<Result<Void>> sendMessages(@RequestBody SendMesDto sendMesDto) {
|
||||||
try {
|
try {
|
||||||
sendMesDto.setSender(RequestHolder.getuserId());
|
// sendMesDto.setSender(RequestHolder.getuserId());
|
||||||
messagesService.sendMessages(sendMesDto);
|
messagesService.sendMessages(sendMesDto);
|
||||||
Result<Void> result = new Result<>();
|
Result<Void> result = new Result<>();
|
||||||
result.setCode(200);
|
result.setCode(200);
|
||||||
|
@ -43,9 +43,9 @@ public class MessagesController {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@GetMapping("/getMessages")
|
// @GetMapping("/getMessages")
|
||||||
public Result<List<historyMessagesVo>> getMessages(@RequestParam("recipient") String recipient) {
|
// public Result<List<historyMessagesVo>> getMessages(@RequestParam("recipient") String recipient) {
|
||||||
List<historyMessagesVo> messages = messagesService.getMessages(RequestHolder.getuserId(), recipient);
|
//// List<historyMessagesVo> messages = messagesService.getMessages(RequestHolder.getuserId(), recipient);
|
||||||
return Result.ok("查询成功", messages);
|
//// return Result.ok("查询成功", messages);
|
||||||
}
|
// }
|
||||||
}
|
}
|
||||||
|
|
|
@ -46,7 +46,7 @@ public class RelationshipController {
|
||||||
@ApiOperation("发送添加请求")
|
@ApiOperation("发送添加请求")
|
||||||
@GetMapping("/addRequest")
|
@GetMapping("/addRequest")
|
||||||
public ResponseEntity<Result<Void>> requestFriend(@RequestParam("username") String username){
|
public ResponseEntity<Result<Void>> requestFriend(@RequestParam("username") String username){
|
||||||
String userId = RequestHolder.getuserId();
|
String userId = RequestHolder.getuserId().toString();
|
||||||
try {
|
try {
|
||||||
relationshipService.requestFriend(userId, username);
|
relationshipService.requestFriend(userId, username);
|
||||||
// 成功返回
|
// 成功返回
|
||||||
|
@ -118,21 +118,21 @@ public class RelationshipController {
|
||||||
*
|
*
|
||||||
* @param relationshipDto
|
* @param relationshipDto
|
||||||
*/
|
*/
|
||||||
@ApiOperation("删除好友")
|
// @ApiOperation("删除好友")
|
||||||
@PostMapping("/removeFriend")
|
// @PostMapping("/removeFriend")
|
||||||
public void removeFriend(@RequestBody RelationshipDto relationshipDto) {
|
// public void removeFriend(@RequestBody RelationshipDto relationshipDto) {
|
||||||
relationshipService.removeFriend(RequestHolder.getuserId(), relationshipDto.getFriend());
|
// relationshipService.removeFriend(RequestHolder.getuserId(), relationshipDto.getFriend());
|
||||||
}
|
// }
|
||||||
|
//
|
||||||
/**
|
// /**
|
||||||
* 获取好友列表
|
// * 获取好友列表
|
||||||
* @return
|
// * @return
|
||||||
*/
|
// */
|
||||||
@ApiOperation("获取好友列表")
|
// @ApiOperation("获取好友列表")
|
||||||
@GetMapping("/getFriends")
|
// @GetMapping("/getFriends")
|
||||||
public List<UserListVo> getFriends() {
|
// public List<UserListVo> getFriends() {
|
||||||
return relationshipService.getFriends(RequestHolder.getuserId());
|
// return relationshipService.getFriends(RequestHolder.getuserId());
|
||||||
}
|
// }
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,22 +2,30 @@ package com.xubx.springboot_01demo.controller;
|
||||||
|
|
||||||
|
|
||||||
import com.alibaba.fastjson.JSONObject;
|
import com.alibaba.fastjson.JSONObject;
|
||||||
|
import com.xubx.springboot_01demo.dto.user.EditMaterialDto;
|
||||||
|
import com.xubx.springboot_01demo.dto.user.NewFavoritesDto;
|
||||||
|
import com.xubx.springboot_01demo.pojo.Blogs;
|
||||||
import com.xubx.springboot_01demo.pojo.User;
|
import com.xubx.springboot_01demo.pojo.User;
|
||||||
|
import com.xubx.springboot_01demo.pojo.UserFavorite;
|
||||||
import com.xubx.springboot_01demo.service.UserService;
|
import com.xubx.springboot_01demo.service.UserService;
|
||||||
import com.xubx.springboot_01demo.utils.token.RequestHolder;
|
import com.xubx.springboot_01demo.utils.token.RequestHolder;
|
||||||
import com.xubx.springboot_01demo.utils.token.TokenGenerate;
|
import com.xubx.springboot_01demo.utils.token.TokenGenerate;
|
||||||
|
import com.xubx.springboot_01demo.vo.getUserInfoVo;
|
||||||
import io.swagger.annotations.Api;
|
import io.swagger.annotations.Api;
|
||||||
import io.swagger.annotations.ApiOperation;
|
import io.swagger.annotations.ApiOperation;
|
||||||
import lombok.extern.slf4j.Slf4j;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
import org.springframework.http.HttpStatus;
|
import org.springframework.http.HttpStatus;
|
||||||
import org.springframework.http.ResponseEntity;
|
import org.springframework.http.ResponseEntity;
|
||||||
|
import org.springframework.validation.BindingResult;
|
||||||
import org.springframework.web.bind.annotation.*;
|
import org.springframework.web.bind.annotation.*;
|
||||||
import org.springframework.web.multipart.MultipartFile;
|
import org.springframework.web.multipart.MultipartFile;
|
||||||
|
|
||||||
import javax.annotation.Resource;
|
import javax.annotation.Resource;
|
||||||
import javax.servlet.http.HttpSession;
|
import javax.servlet.http.HttpSession;
|
||||||
|
import javax.validation.Valid;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
@RestController //注解标识这是一个控制器类
|
@RestController //注解标识这是一个控制器类
|
||||||
@CrossOrigin //加上CrossOrigin可解决跨域问题
|
@CrossOrigin //加上CrossOrigin可解决跨域问题
|
||||||
|
@ -28,8 +36,6 @@ public class UserController {
|
||||||
@Resource
|
@Resource
|
||||||
UserService userService;
|
UserService userService;
|
||||||
// 注入session
|
// 注入session
|
||||||
@Resource
|
|
||||||
private HttpSession session;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 用户注册
|
* 用户注册
|
||||||
|
@ -67,16 +73,17 @@ public class UserController {
|
||||||
*/
|
*/
|
||||||
@PostMapping("/login")
|
@PostMapping("/login")
|
||||||
@ApiOperation("用户登录")
|
@ApiOperation("用户登录")
|
||||||
public String login(@RequestBody User user) {
|
public ResponseEntity<String> login(@RequestBody User user) {
|
||||||
log.info("用户登录:{}", user.getUsername());
|
log.info("用户登录:{}", user);
|
||||||
//登陆
|
|
||||||
if (userService.findUserByUsername(user)) {
|
// 调用 Service 中的登录方法
|
||||||
String token = new TokenGenerate().generateToken(user.getUsername());
|
String token = userService.login(user);
|
||||||
//将username存入session
|
|
||||||
session.setAttribute("username", user.getUsername());
|
if (token != null) {
|
||||||
return token;
|
return ResponseEntity.ok(token);
|
||||||
|
} else {
|
||||||
|
return ResponseEntity.status(HttpStatus.UNAUTHORIZED).body("登录失败,用户名或密码错误");
|
||||||
}
|
}
|
||||||
return "false";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -85,9 +92,9 @@ public class UserController {
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
@GetMapping("/getUsername")
|
@GetMapping("/getUsername")
|
||||||
@ApiOperation("获取用户名")
|
@ApiOperation("获取当前用户名")
|
||||||
public String getUserName() {
|
public String getUserName() {
|
||||||
return RequestHolder.getuserId();
|
return userService.getUser("").getUsername();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -95,22 +102,22 @@ public class UserController {
|
||||||
*
|
*
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
@RequestMapping("/uploadAvatar")
|
// @RequestMapping("/uploadAvatar")
|
||||||
@ApiOperation("上传头像")
|
// @ApiOperation("上传头像")
|
||||||
public void uploadAvatar(MultipartFile file) throws IOException {
|
// public void uploadAvatar(MultipartFile file) throws IOException {
|
||||||
String pType = file.getContentType();
|
// String pType = file.getContentType();
|
||||||
pType = pType.substring(pType.indexOf("/") + 1);
|
// pType = pType.substring(pType.indexOf("/") + 1);
|
||||||
if ("jpeg".equals(pType)) {
|
// if ("jpeg".equals(pType)) {
|
||||||
pType = "jpg";
|
// pType = "jpg";
|
||||||
}
|
// }
|
||||||
long time = System.currentTimeMillis();
|
// long time = System.currentTimeMillis();
|
||||||
String currentWorkingDirectory = System.getProperty("user.dir");
|
// String currentWorkingDirectory = System.getProperty("user.dir");
|
||||||
String relativePath = "/images/avatar/" + time + "." + pType;
|
// String relativePath = "/images/avatar/" + time + "." + pType;
|
||||||
String absolutePath = currentWorkingDirectory + "/static" + relativePath;
|
// String absolutePath = currentWorkingDirectory + "/static" + relativePath;
|
||||||
file.transferTo(new File(absolutePath));
|
// file.transferTo(new File(absolutePath));
|
||||||
System.out.println("导入数据库的路径:" + relativePath + "当前用户:" + RequestHolder.getuserId());
|
// System.out.println("导入数据库的路径:" + relativePath + "当前用户Id:" + RequestHolder.getuserId());
|
||||||
userService.addAvatar(relativePath, RequestHolder.getuserId());
|
// userService.addAvatar(relativePath, RequestHolder.getuserId());
|
||||||
}
|
// }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 获取头像
|
* 获取头像
|
||||||
|
@ -120,9 +127,7 @@ public class UserController {
|
||||||
@GetMapping("/getAvatar")
|
@GetMapping("/getAvatar")
|
||||||
@ApiOperation("获取头像")
|
@ApiOperation("获取头像")
|
||||||
public String getAvatar() {
|
public String getAvatar() {
|
||||||
String path = userService.getAvatar(RequestHolder.getuserId());
|
return userService.getUser("").getAvatar();
|
||||||
System.out.println("发给前端的路径:" + path);
|
|
||||||
return path;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -134,11 +139,14 @@ public class UserController {
|
||||||
@GetMapping("usernameChange")
|
@GetMapping("usernameChange")
|
||||||
@ApiOperation("修改用户名")
|
@ApiOperation("修改用户名")
|
||||||
public ResponseEntity<String> usernameChange(@RequestParam("username") String username) {
|
public ResponseEntity<String> usernameChange(@RequestParam("username") String username) {
|
||||||
if (userService.usernameChange(RequestHolder.getuserId(), username)) {
|
log.info("修改用户名:{}", username);
|
||||||
RequestHolder.add(username);
|
|
||||||
|
try {
|
||||||
|
userService.usernameChange(username);
|
||||||
return ResponseEntity.ok("修改成功!");
|
return ResponseEntity.ok("修改成功!");
|
||||||
|
} catch (Exception e) {
|
||||||
|
return ResponseEntity.ok("修改失败!");
|
||||||
}
|
}
|
||||||
return ResponseEntity.ok("该用户已存在!");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -148,22 +156,119 @@ public class UserController {
|
||||||
* @param newPassword
|
* @param newPassword
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
@GetMapping("passwordChange")
|
// @GetMapping("passwordChange")
|
||||||
@ApiOperation("修改密码")
|
// @ApiOperation("修改密码")
|
||||||
public ResponseEntity<String> passwordChange(@RequestParam("oldPassword") String oldPassword, @RequestParam("newPassword") String newPassword) {
|
// public ResponseEntity<String> passwordChange(@RequestParam("oldPassword") String oldPassword, @RequestParam("newPassword") String newPassword) {
|
||||||
if (userService.passwordChange(RequestHolder.getuserId(), oldPassword, newPassword)) {
|
// if (userService.passwordChange(RequestHolder.getuserId(), oldPassword, newPassword)) {
|
||||||
return ResponseEntity.ok("修改成功!");
|
// return ResponseEntity.ok("修改成功!");
|
||||||
|
// }
|
||||||
|
// return ResponseEntity.ok("原密码输入错误!");
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * 查看是否有好友请求
|
||||||
|
// *
|
||||||
|
// * @return
|
||||||
|
// */
|
||||||
|
// @GetMapping("/checkFriendRequest")
|
||||||
|
// public JSONObject haveFriendRequest() {
|
||||||
|
// return userService.checkFriendRequest(RequestHolder.getuserId());
|
||||||
|
// }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取用户信息
|
||||||
|
*
|
||||||
|
* @return {@link ResponseEntity }<{@link getUserInfoVo }>
|
||||||
|
*/
|
||||||
|
@GetMapping("/getUserInfo")
|
||||||
|
@ApiOperation("获取用户信息")
|
||||||
|
public ResponseEntity<getUserInfoVo> getUserInfo(@RequestParam("userId") String userId) {
|
||||||
|
log.info("获取用户信息:{}", userId);
|
||||||
|
|
||||||
|
try {
|
||||||
|
return ResponseEntity.ok(userService.getUserInfo(userId));
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.info("获取用户信息失败", e);
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body(null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping("/editMaterial")
|
||||||
|
@ApiOperation("修改用户资料")
|
||||||
|
public ResponseEntity<String> editMaterial(@RequestBody EditMaterialDto editMaterialDto) {
|
||||||
|
log.info("修改用户资料");
|
||||||
|
try {
|
||||||
|
userService.editMaterial(editMaterialDto);
|
||||||
|
return ResponseEntity.ok("修改成功!");
|
||||||
|
} catch (Exception e) {
|
||||||
|
return ResponseEntity.ok("修改失败!");
|
||||||
}
|
}
|
||||||
return ResponseEntity.ok("原密码输入错误!");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 查看是否有好友请求
|
* 新建收藏夹
|
||||||
*
|
*
|
||||||
* @return
|
* @param newFavorites
|
||||||
|
* @return {@link ResponseEntity }<{@link String }>
|
||||||
*/
|
*/
|
||||||
@GetMapping("/checkFriendRequest")
|
@PostMapping("/newFavorites")
|
||||||
public JSONObject haveFriendRequest() {
|
@ApiOperation("新建收藏夹")
|
||||||
return userService.checkFriendRequest(RequestHolder.getuserId());
|
public ResponseEntity<String> newFavorites(@Valid @RequestBody NewFavoritesDto newFavorites, BindingResult bindingResult) {
|
||||||
|
log.info("新建收藏夹,入参:{}", JSONObject.toJSONString(newFavorites));
|
||||||
|
|
||||||
|
if (bindingResult.hasErrors()) {
|
||||||
|
return ResponseEntity.status(HttpStatus.BAD_REQUEST).body(bindingResult.getFieldError().getDefaultMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
Integer userId = RequestHolder.getuserId();
|
||||||
|
if (userId == null) {
|
||||||
|
return ResponseEntity.status(HttpStatus.UNAUTHORIZED).body("用户未登录");
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
String result = userService.newFavorites(newFavorites);
|
||||||
|
if (result != null) {
|
||||||
|
return ResponseEntity.status(HttpStatus.BAD_REQUEST).body(result);
|
||||||
|
}
|
||||||
|
return ResponseEntity.ok("收藏夹创建成功");
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("新建收藏夹失败", e);
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body("新建收藏夹失败");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取用户收藏夹列表
|
||||||
|
* @return {@link ResponseEntity }<{@link List }<{@link UserFavorite }>>
|
||||||
|
*/
|
||||||
|
@GetMapping("/getFavorites")
|
||||||
|
@ApiOperation("获取收藏夹列表")
|
||||||
|
public ResponseEntity<List<UserFavorite>> getUserFavorites(@RequestParam("userId") String userId) {
|
||||||
|
log.info("获取收藏夹列表");
|
||||||
|
|
||||||
|
try {
|
||||||
|
return ResponseEntity.ok(userService.getUserFavorites(userId));
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("获取收藏夹列表失败", e);
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body(null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取收藏夹内的博客
|
||||||
|
* @param favoriteId
|
||||||
|
* @return {@link ResponseEntity }<{@link List }<{@link Blogs }>>
|
||||||
|
*/
|
||||||
|
@GetMapping("/getBlogsByFavoriteId")
|
||||||
|
@ApiOperation("根据收藏夹id获取收藏夹内的博客")
|
||||||
|
public ResponseEntity<?> getBlogsByFavoriteId(@RequestParam("favoriteId") String favoriteId) {
|
||||||
|
log.info("根据收藏夹id获取收藏夹内的博客: {}", favoriteId);
|
||||||
|
|
||||||
|
try {
|
||||||
|
return ResponseEntity.ok(userService.getBlogsByFavoriteId(favoriteId));
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("根据收藏夹id获取收藏夹内的博客失败", e);
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body("根据收藏夹id获取收藏夹内的博客失败");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,7 +12,7 @@ import javax.validation.constraints.NotBlank;
|
||||||
* @date 2024/11/01
|
* @date 2024/11/01
|
||||||
*/
|
*/
|
||||||
@Data
|
@Data
|
||||||
public class addBlogDto {
|
public class AddBlogDto {
|
||||||
@NotBlank
|
@NotBlank
|
||||||
private String title;
|
private String title;
|
||||||
|
|
|
@ -0,0 +1,19 @@
|
||||||
|
package com.xubx.springboot_01demo.dto.blog;
|
||||||
|
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
import javax.validation.constraints.NotBlank;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 收藏博客入参
|
||||||
|
* @author Xubx
|
||||||
|
* @date 2024/11/03
|
||||||
|
*/
|
||||||
|
@Data
|
||||||
|
public class CollectBlogDto {
|
||||||
|
@NotBlank
|
||||||
|
private Integer blogId;
|
||||||
|
|
||||||
|
@NotBlank
|
||||||
|
private Integer favoriteId;
|
||||||
|
}
|
|
@ -0,0 +1,23 @@
|
||||||
|
package com.xubx.springboot_01demo.dto.blog;
|
||||||
|
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
import javax.validation.constraints.NotBlank;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取用户博客入参
|
||||||
|
* @author Xubx
|
||||||
|
* @date 2024/11/03
|
||||||
|
*/
|
||||||
|
@Data
|
||||||
|
public class GetUserBlogsDto {
|
||||||
|
@NotBlank
|
||||||
|
private String userId;
|
||||||
|
|
||||||
|
@NotBlank
|
||||||
|
private int page;
|
||||||
|
|
||||||
|
@NotBlank
|
||||||
|
private int pageSize;
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,20 @@
|
||||||
|
package com.xubx.springboot_01demo.dto.user;
|
||||||
|
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 修改用户资料入参
|
||||||
|
* @author Xubx
|
||||||
|
* @date 2024/11/02
|
||||||
|
*/
|
||||||
|
@Data
|
||||||
|
public class EditMaterialDto {
|
||||||
|
private String username;
|
||||||
|
|
||||||
|
private String avatar;
|
||||||
|
|
||||||
|
private String gender;
|
||||||
|
|
||||||
|
private String bio;
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,26 @@
|
||||||
|
package com.xubx.springboot_01demo.dto.user;
|
||||||
|
|
||||||
|
import io.swagger.annotations.ApiModel;
|
||||||
|
import io.swagger.annotations.ApiModelProperty;
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
import javax.validation.constraints.NotBlank;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 新建收藏夹入参
|
||||||
|
* @author Xubx
|
||||||
|
* @date 2024/11/03
|
||||||
|
*/
|
||||||
|
@Data
|
||||||
|
@ApiModel("新建收藏夹入参")
|
||||||
|
public class NewFavoritesDto {
|
||||||
|
@NotBlank
|
||||||
|
@ApiModelProperty("收藏夹名称")
|
||||||
|
private String name;
|
||||||
|
|
||||||
|
@ApiModelProperty("收藏夹描述")
|
||||||
|
private String description;
|
||||||
|
|
||||||
|
@ApiModelProperty("收藏夹状态:默认是0 公开,1 私密")
|
||||||
|
private Integer favoriteStatus;
|
||||||
|
}
|
|
@ -0,0 +1,7 @@
|
||||||
|
package com.xubx.springboot_01demo.mapper;
|
||||||
|
|
||||||
|
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||||
|
import com.xubx.springboot_01demo.pojo.BlogLike;
|
||||||
|
|
||||||
|
public interface BlogLikeMapper extends BaseMapper<BlogLike> {
|
||||||
|
}
|
|
@ -1,7 +1,8 @@
|
||||||
package com.xubx.springboot_01demo.mapper;
|
package com.xubx.springboot_01demo.mapper;
|
||||||
|
|
||||||
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||||
import com.xubx.springboot_01demo.dto.blog.addBlogDto;
|
import com.xubx.springboot_01demo.dto.blog.AddBlogDto;
|
||||||
|
import com.xubx.springboot_01demo.dto.blog.CollectBlogDto;
|
||||||
import com.xubx.springboot_01demo.pojo.Blogs;
|
import com.xubx.springboot_01demo.pojo.Blogs;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
@ -12,7 +13,7 @@ public interface BlogsMapper extends BaseMapper<Blogs> {
|
||||||
//根据id获取博客
|
//根据id获取博客
|
||||||
Blogs findByIdBlogs(int id);
|
Blogs findByIdBlogs(int id);
|
||||||
//新增博客
|
//新增博客
|
||||||
void addBlogs(addBlogDto blogs);
|
void addBlogs(AddBlogDto blogs);
|
||||||
//更新博客
|
//更新博客
|
||||||
void updateBlogs(Blogs blogs);
|
void updateBlogs(Blogs blogs);
|
||||||
//删除博客
|
//删除博客
|
||||||
|
@ -24,4 +25,105 @@ public interface BlogsMapper extends BaseMapper<Blogs> {
|
||||||
* @param categoryId
|
* @param categoryId
|
||||||
*/
|
*/
|
||||||
void insertCategory(int blogId, String categoryId);
|
void insertCategory(int blogId, String categoryId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取该用户总访问量
|
||||||
|
* @param userId
|
||||||
|
* @return {@link Integer }
|
||||||
|
*/
|
||||||
|
String getTotalViewCount(Integer userId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取该用户总博客数
|
||||||
|
* @param userId
|
||||||
|
* @return {@link Integer }
|
||||||
|
*/
|
||||||
|
String getTotalBlogCount(Integer userId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取该用户被收藏总数
|
||||||
|
* @param userId
|
||||||
|
* @return {@link Integer }
|
||||||
|
*/
|
||||||
|
String getTotalCollectCount(Integer userId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取该用户被点赞总数
|
||||||
|
* @param userId
|
||||||
|
* @return {@link Integer }
|
||||||
|
*/
|
||||||
|
String getTotalLikeCount(Integer userId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取该用户被评论总数
|
||||||
|
* @param userId
|
||||||
|
* @return {@link Integer }
|
||||||
|
*/
|
||||||
|
String getTotalCommentCount(Integer userId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取该用户被分享总数
|
||||||
|
* @param userId
|
||||||
|
* @return {@link Integer }
|
||||||
|
*/
|
||||||
|
String getTotalShareCount(Integer userId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取某个博客的该用户的点赞状态
|
||||||
|
* @param blogId
|
||||||
|
* @param userId
|
||||||
|
* @return {@link Integer }
|
||||||
|
*/
|
||||||
|
Integer getLikeStatus(int blogId, Integer userId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取某个博客的该用户的收藏状态
|
||||||
|
* @param blogId
|
||||||
|
* @param userId
|
||||||
|
* @return {@link Integer }
|
||||||
|
*/
|
||||||
|
Integer getCollectStatus(int blogId, Integer userId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取某个博客的点赞数
|
||||||
|
* @param blogId
|
||||||
|
* @return {@link Integer }
|
||||||
|
*/
|
||||||
|
String getLikeCount(int blogId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取某个博客的浏览数
|
||||||
|
* @param blogId
|
||||||
|
* @return {@link Object }
|
||||||
|
*/
|
||||||
|
String getViewCount(int blogId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取某个博客的收藏数
|
||||||
|
* @param blogId
|
||||||
|
* @return {@link Object }
|
||||||
|
*/
|
||||||
|
String getCollectCount(int blogId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取某个博客的评论数
|
||||||
|
* @param blogId
|
||||||
|
* @return {@link Object }
|
||||||
|
*/
|
||||||
|
String getCommentCount(int blogId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 当前用户进行博客的收藏
|
||||||
|
* @param collectBlogDto
|
||||||
|
*/
|
||||||
|
void collectBlog(CollectBlogDto collectBlogDto);
|
||||||
|
|
||||||
|
List<Blogs> getBlogsByFavoriteId(String favoriteId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取该博客的所有点赞用户id
|
||||||
|
* @param blogId
|
||||||
|
* @return {@link List }<{@link String }>
|
||||||
|
*/
|
||||||
|
List<String> getUserIdsByBlogId(String blogId);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,12 +1,13 @@
|
||||||
package com.xubx.springboot_01demo.mapper;
|
package com.xubx.springboot_01demo.mapper;
|
||||||
|
|
||||||
|
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||||
import com.xubx.springboot_01demo.pojo.Relationship;
|
import com.xubx.springboot_01demo.pojo.Relationship;
|
||||||
import com.xubx.springboot_01demo.pojo.User;
|
import com.xubx.springboot_01demo.pojo.User;
|
||||||
import com.xubx.springboot_01demo.vo.UserListVo;
|
import com.xubx.springboot_01demo.vo.UserListVo;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
public interface RelationshipMapper {
|
public interface RelationshipMapper extends BaseMapper<Relationship> {
|
||||||
//1. 查询用户接口
|
//1. 查询用户接口
|
||||||
List<User> findUserByUsername(String currentName,String username);
|
List<User> findUserByUsername(String currentName,String username);
|
||||||
//2.添加用户接口
|
//2.添加用户接口
|
||||||
|
@ -19,4 +20,11 @@ public interface RelationshipMapper {
|
||||||
List<UserListVo> getFriends(String username);
|
List<UserListVo> getFriends(String username);
|
||||||
//6.查看是否有好友请求
|
//6.查看是否有好友请求
|
||||||
List<Relationship> checkFriendRequest(String getuserId);
|
List<Relationship> checkFriendRequest(String getuserId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取粉丝数量
|
||||||
|
* @param userId
|
||||||
|
* @return {@link Integer }
|
||||||
|
*/
|
||||||
|
String getTotalFollowerCount(Integer userId);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,7 @@
|
||||||
|
package com.xubx.springboot_01demo.mapper;
|
||||||
|
|
||||||
|
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||||
|
import com.xubx.springboot_01demo.pojo.UserFavoriteBlog;
|
||||||
|
|
||||||
|
public interface UserFavoriteBlogMapper extends BaseMapper<UserFavoriteBlog> {
|
||||||
|
}
|
|
@ -0,0 +1,7 @@
|
||||||
|
package com.xubx.springboot_01demo.mapper;
|
||||||
|
|
||||||
|
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||||
|
import com.xubx.springboot_01demo.pojo.UserFavorite;
|
||||||
|
|
||||||
|
public interface UserFavoriteMapper extends BaseMapper<UserFavorite> {
|
||||||
|
}
|
|
@ -1,10 +1,9 @@
|
||||||
package com.xubx.springboot_01demo.mapper;
|
package com.xubx.springboot_01demo.mapper;
|
||||||
|
|
||||||
|
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
||||||
import com.xubx.springboot_01demo.pojo.User;
|
import com.xubx.springboot_01demo.pojo.User;
|
||||||
|
|
||||||
public interface UserMapper {
|
public interface UserMapper extends BaseMapper<User> {
|
||||||
//1. 查询用户接口
|
|
||||||
User findUserByUsername(String username);
|
|
||||||
//2.添加用户接口
|
//2.添加用户接口
|
||||||
void insertUser(User user);
|
void insertUser(User user);
|
||||||
//3.添加头像接口
|
//3.添加头像接口
|
||||||
|
|
|
@ -0,0 +1,24 @@
|
||||||
|
package com.xubx.springboot_01demo.pojo;
|
||||||
|
|
||||||
|
import com.baomidou.mybatisplus.annotation.TableField;
|
||||||
|
import com.baomidou.mybatisplus.annotation.TableName;
|
||||||
|
import io.swagger.annotations.ApiModel;
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
import java.sql.Timestamp;
|
||||||
|
|
||||||
|
@Data
|
||||||
|
@TableName("blog_like")
|
||||||
|
public class BlogLike {
|
||||||
|
@TableField("id")
|
||||||
|
private int id;
|
||||||
|
|
||||||
|
@TableField("blog_id")
|
||||||
|
private Integer blogId;
|
||||||
|
|
||||||
|
@TableField("user_id")
|
||||||
|
private Integer userId;
|
||||||
|
|
||||||
|
@TableField("created_time")
|
||||||
|
private Timestamp createdTime;
|
||||||
|
}
|
|
@ -25,7 +25,7 @@ public class Blogs implements Serializable {
|
||||||
*/
|
*/
|
||||||
@ApiModelProperty(value = "id")
|
@ApiModelProperty(value = "id")
|
||||||
@TableId(type = IdType.AUTO)
|
@TableId(type = IdType.AUTO)
|
||||||
private int id;
|
private Integer id;
|
||||||
|
|
||||||
@ApiModelProperty(value = "标题")
|
@ApiModelProperty(value = "标题")
|
||||||
private String title;
|
private String title;
|
||||||
|
@ -40,7 +40,7 @@ public class Blogs implements Serializable {
|
||||||
private String coverImage;
|
private String coverImage;
|
||||||
|
|
||||||
@ApiModelProperty(value = "作者id")
|
@ApiModelProperty(value = "作者id")
|
||||||
private String authorId;
|
private Integer authorId;
|
||||||
|
|
||||||
@ApiModelProperty(value = "点赞数")
|
@ApiModelProperty(value = "点赞数")
|
||||||
private int likeCount;
|
private int likeCount;
|
||||||
|
@ -48,8 +48,11 @@ public class Blogs implements Serializable {
|
||||||
@ApiModelProperty(value = "浏览数")
|
@ApiModelProperty(value = "浏览数")
|
||||||
private int viewCount;
|
private int viewCount;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "评论数")
|
||||||
|
private int commentCount;
|
||||||
|
|
||||||
@ApiModelProperty(value = "收藏数")
|
@ApiModelProperty(value = "收藏数")
|
||||||
private int favoriteCount;
|
private int collectCount;
|
||||||
|
|
||||||
@ApiModelProperty(value = "分享数")
|
@ApiModelProperty(value = "分享数")
|
||||||
private int shareCount;
|
private int shareCount;
|
||||||
|
|
|
@ -3,7 +3,7 @@ package com.xubx.springboot_01demo.pojo;
|
||||||
import java.sql.Timestamp;
|
import java.sql.Timestamp;
|
||||||
|
|
||||||
public class Relationship {
|
public class Relationship {
|
||||||
private int id;
|
private Integer id;
|
||||||
/**用户名*/
|
/**用户名*/
|
||||||
private String username;
|
private String username;
|
||||||
/**好友*/
|
/**好友*/
|
||||||
|
|
|
@ -2,70 +2,52 @@ package com.xubx.springboot_01demo.pojo;
|
||||||
|
|
||||||
|
|
||||||
import com.baomidou.mybatisplus.annotation.TableField;
|
import com.baomidou.mybatisplus.annotation.TableField;
|
||||||
|
import com.baomidou.mybatisplus.annotation.TableName;
|
||||||
|
import io.swagger.annotations.ApiModel;
|
||||||
|
import io.swagger.annotations.ApiModelProperty;
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
import java.io.Serializable;
|
import java.io.Serializable;
|
||||||
|
import java.sql.Timestamp;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 用户实体类
|
* 用户实体类
|
||||||
*/
|
*/
|
||||||
|
@Data
|
||||||
|
@ApiModel
|
||||||
|
@TableName("user")
|
||||||
public class User implements Serializable {
|
public class User implements Serializable {
|
||||||
private int id;
|
private Integer id;
|
||||||
/** 用户名*/
|
|
||||||
|
@ApiModelProperty(value = "用户名")
|
||||||
private String username;
|
private String username;
|
||||||
/** 密码*/
|
|
||||||
|
@ApiModelProperty(value = "密码")
|
||||||
private String password;
|
private String password;
|
||||||
/** 头像*/
|
|
||||||
|
@ApiModelProperty(value = "头像")
|
||||||
|
@TableField("avatar")
|
||||||
private String avatar;
|
private String avatar;
|
||||||
/** 用户状态*/
|
|
||||||
private int state;
|
|
||||||
|
|
||||||
@Override
|
@ApiModelProperty(value = "用户状态")
|
||||||
public String toString() {
|
private int status;
|
||||||
return "User{" +
|
|
||||||
"username='" + username + '\'' +
|
|
||||||
", password='" + password + '\'' +
|
|
||||||
", avatar='" + avatar + '\'' +
|
|
||||||
", state=" + state +
|
|
||||||
'}';
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getId() {
|
@ApiModelProperty(value = "用户邮箱")
|
||||||
return id;
|
private String email;
|
||||||
}
|
|
||||||
|
|
||||||
public void setId(int id) {
|
@ApiModelProperty(value = "用户性别")
|
||||||
this.id = id;
|
private String gender;
|
||||||
}
|
|
||||||
|
|
||||||
public String getUsername() {
|
@ApiModelProperty(value = "个人简介")
|
||||||
return username;
|
@TableField("bio")
|
||||||
}
|
private String bio;
|
||||||
|
|
||||||
public void setUsername(String username) {
|
@ApiModelProperty(value = "创建时间")
|
||||||
this.username = username;
|
@TableField("created_time")
|
||||||
}
|
private Timestamp createdTime;
|
||||||
|
|
||||||
public String getPassword() {
|
@ApiModelProperty(value = "更新时间")
|
||||||
return password;
|
@TableField("updated_time")
|
||||||
}
|
private Timestamp updateTime;
|
||||||
|
|
||||||
public void setPassword(String password) {
|
|
||||||
this.password = password;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getAvatar() {
|
|
||||||
return avatar;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setAvatar(String avatar) {
|
|
||||||
this.avatar = avatar;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getState() {
|
|
||||||
return state;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setState(int state) {
|
|
||||||
this.state = state;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,41 @@
|
||||||
|
package com.xubx.springboot_01demo.pojo;
|
||||||
|
|
||||||
|
import com.baomidou.mybatisplus.annotation.TableField;
|
||||||
|
import com.baomidou.mybatisplus.annotation.TableName;
|
||||||
|
import io.swagger.annotations.ApiModel;
|
||||||
|
import io.swagger.annotations.ApiModelProperty;
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.sql.Timestamp;
|
||||||
|
|
||||||
|
@Data
|
||||||
|
@ApiModel(value = "UserFavorite", description = "用户收藏")
|
||||||
|
@TableName("user_favorite")
|
||||||
|
public class UserFavorite implements Serializable {
|
||||||
|
private int id;
|
||||||
|
|
||||||
|
@TableField("name")
|
||||||
|
@ApiModelProperty(value = "收藏名称")
|
||||||
|
private String name;
|
||||||
|
|
||||||
|
@TableField("description")
|
||||||
|
@ApiModelProperty(value = "收藏描述")
|
||||||
|
private String description;
|
||||||
|
|
||||||
|
@TableField("blog_count")
|
||||||
|
@ApiModelProperty(value = "收藏夹内博客数量")
|
||||||
|
private Integer blogCount;
|
||||||
|
|
||||||
|
@TableField("user_id")
|
||||||
|
@ApiModelProperty(value = "用户id")
|
||||||
|
private Integer userId;
|
||||||
|
|
||||||
|
@TableField("favorite_status")
|
||||||
|
@ApiModelProperty(value = "收藏状态: 默认是公开,可选私密")
|
||||||
|
private String favoriteStatus;
|
||||||
|
|
||||||
|
@TableField("created_time")
|
||||||
|
@ApiModelProperty(value = "创建时间")
|
||||||
|
private Timestamp createdTime;
|
||||||
|
}
|
|
@ -0,0 +1,20 @@
|
||||||
|
package com.xubx.springboot_01demo.pojo;
|
||||||
|
|
||||||
|
import com.baomidou.mybatisplus.annotation.TableField;
|
||||||
|
import com.baomidou.mybatisplus.annotation.TableName;
|
||||||
|
import io.swagger.annotations.ApiModel;
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
@Data
|
||||||
|
@ApiModel(value = "UserFavorite", description = "用户收藏夹博客中间表")
|
||||||
|
@TableName("user_favorite_blog")
|
||||||
|
public class UserFavoriteBlog {
|
||||||
|
private int id;
|
||||||
|
|
||||||
|
@TableField("favorite_id")
|
||||||
|
private Integer favoriteId;
|
||||||
|
|
||||||
|
@TableField("blog_id")
|
||||||
|
private Integer blogId;
|
||||||
|
|
||||||
|
}
|
|
@ -1,7 +1,13 @@
|
||||||
package com.xubx.springboot_01demo.service;
|
package com.xubx.springboot_01demo.service;
|
||||||
|
|
||||||
import com.xubx.springboot_01demo.dto.blog.addBlogDto;
|
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
||||||
|
import com.fasterxml.jackson.core.JsonProcessingException;
|
||||||
|
import com.xubx.springboot_01demo.dto.blog.AddBlogDto;
|
||||||
|
import com.xubx.springboot_01demo.dto.blog.CollectBlogDto;
|
||||||
|
import com.xubx.springboot_01demo.dto.blog.GetUserBlogsDto;
|
||||||
|
import com.xubx.springboot_01demo.dto.user.NewFavoritesDto;
|
||||||
import com.xubx.springboot_01demo.pojo.Blogs;
|
import com.xubx.springboot_01demo.pojo.Blogs;
|
||||||
|
import com.xubx.springboot_01demo.vo.GetBlogDetailVo;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
|
@ -10,11 +16,20 @@ public interface BlogService {
|
||||||
List<Blogs> findAllBlogs();
|
List<Blogs> findAllBlogs();
|
||||||
|
|
||||||
//根据id获取博客
|
//根据id获取博客
|
||||||
Blogs findByIdBlogs(int id);
|
GetBlogDetailVo findByIdBlogs(int blogId);
|
||||||
//添加博客
|
//添加博客
|
||||||
void addBlogs(addBlogDto blogs);
|
void addBlogs(AddBlogDto blogs);
|
||||||
//更新博客
|
//更新博客
|
||||||
void updateBlogs(Blogs blogs);
|
void updateBlogs(Blogs blogs);
|
||||||
//删除博客
|
//删除博客
|
||||||
void deleteBlogs(int id);
|
void deleteBlogs(int id);
|
||||||
|
|
||||||
|
Page<Blogs> findBlogsByUserId(GetUserBlogsDto getUserBlogsDto);
|
||||||
|
|
||||||
|
void viewBlog(int blogId);
|
||||||
|
|
||||||
|
void likeBlog(int blogId);
|
||||||
|
|
||||||
|
void collectBlog(CollectBlogDto collectBlogDto) throws JsonProcessingException;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,25 +1,58 @@
|
||||||
package com.xubx.springboot_01demo.service;
|
package com.xubx.springboot_01demo.service;
|
||||||
|
|
||||||
import com.alibaba.fastjson.JSONObject;
|
import com.alibaba.fastjson.JSONObject;
|
||||||
import com.xubx.springboot_01demo.pojo.Relationship;
|
import com.fasterxml.jackson.core.JsonProcessingException;
|
||||||
|
import com.xubx.springboot_01demo.dto.user.EditMaterialDto;
|
||||||
|
import com.xubx.springboot_01demo.dto.user.NewFavoritesDto;
|
||||||
|
import com.xubx.springboot_01demo.pojo.Blogs;
|
||||||
import com.xubx.springboot_01demo.pojo.User;
|
import com.xubx.springboot_01demo.pojo.User;
|
||||||
import com.xubx.springboot_01demo.utils.api.Result;
|
import com.xubx.springboot_01demo.pojo.UserFavorite;
|
||||||
|
import com.xubx.springboot_01demo.vo.getUserInfoVo;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
public interface UserService {
|
public interface UserService {
|
||||||
//登陆判断
|
//登陆判断
|
||||||
Boolean findUserByUsername(User user);
|
String login(User user);
|
||||||
//新增用户
|
//新增用户
|
||||||
Boolean insertUser(User user);
|
Boolean insertUser(User user);
|
||||||
//添加头像
|
//添加头像
|
||||||
void addAvatar(String path,String username);
|
void addAvatar(String path,String userId);
|
||||||
//获取头像地址
|
//获取头像地址
|
||||||
String getAvatar(String name);
|
String getAvatar(String name);
|
||||||
//5.修改用户名
|
//5.修改用户名
|
||||||
boolean usernameChange(String usernameNow,String username);
|
void usernameChange(String username);
|
||||||
//6.修改密码
|
//6.修改密码
|
||||||
boolean passwordChange(String username,String oldPassword,String newPassword);
|
boolean passwordChange(String username,String oldPassword,String newPassword);
|
||||||
|
|
||||||
JSONObject checkFriendRequest(String getuserId);
|
JSONObject checkFriendRequest(String getuserId);
|
||||||
|
|
||||||
|
User getUser(String getUserId);
|
||||||
|
|
||||||
|
getUserInfoVo getUserInfo(String getuserId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
*修改用户信息
|
||||||
|
*/
|
||||||
|
void editMaterial(EditMaterialDto editMaterialDto);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 新建收藏夹
|
||||||
|
* @param newFavorites
|
||||||
|
*/
|
||||||
|
String newFavorites(NewFavoritesDto newFavorites);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取用户的收藏夹
|
||||||
|
* @param userId
|
||||||
|
* @return {@link List }<{@link UserFavorite }>
|
||||||
|
*/
|
||||||
|
List<UserFavorite> getUserFavorites(String userId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 根据收藏夹id获取收藏夹中的博客
|
||||||
|
* @param favoriteId
|
||||||
|
* @return {@link List }<{@link Blogs }>
|
||||||
|
*/
|
||||||
|
List<Blogs> getBlogsByFavoriteId(String favoriteId) throws JsonProcessingException;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,33 +1,155 @@
|
||||||
package com.xubx.springboot_01demo.service.impl;
|
package com.xubx.springboot_01demo.service.impl;
|
||||||
|
|
||||||
import com.xubx.springboot_01demo.dto.blog.addBlogDto;
|
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
||||||
|
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
||||||
|
import com.fasterxml.jackson.core.JsonProcessingException;
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
|
import com.xubx.springboot_01demo.configuration.constantConfiguration;
|
||||||
|
import com.xubx.springboot_01demo.dto.blog.AddBlogDto;
|
||||||
|
import com.xubx.springboot_01demo.dto.blog.CollectBlogDto;
|
||||||
|
import com.xubx.springboot_01demo.dto.blog.GetUserBlogsDto;
|
||||||
import com.xubx.springboot_01demo.mapper.BlogsMapper;
|
import com.xubx.springboot_01demo.mapper.BlogsMapper;
|
||||||
import com.xubx.springboot_01demo.pojo.Blogs;
|
import com.xubx.springboot_01demo.pojo.Blogs;
|
||||||
|
import com.xubx.springboot_01demo.pojo.User;
|
||||||
import com.xubx.springboot_01demo.service.BlogService;
|
import com.xubx.springboot_01demo.service.BlogService;
|
||||||
|
import com.xubx.springboot_01demo.service.UserService;
|
||||||
import com.xubx.springboot_01demo.utils.token.RequestHolder;
|
import com.xubx.springboot_01demo.utils.token.RequestHolder;
|
||||||
|
import com.xubx.springboot_01demo.vo.GetBlogDetailVo;
|
||||||
|
import lombok.extern.slf4j.Slf4j;
|
||||||
|
import org.springframework.beans.BeanUtils;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.data.redis.core.RedisTemplate;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
import javax.annotation.Resource;
|
import javax.annotation.Resource;
|
||||||
import java.sql.Timestamp;
|
import java.sql.Timestamp;
|
||||||
import java.time.LocalDateTime;
|
import java.time.LocalDateTime;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
@Service
|
@Service
|
||||||
|
@Slf4j
|
||||||
public class BlogsServiceImpl implements BlogService {
|
public class BlogsServiceImpl implements BlogService {
|
||||||
@Resource
|
@Resource
|
||||||
BlogsMapper blogsMapper;
|
BlogsMapper blogsMapper;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
UserService userService;
|
||||||
|
|
||||||
|
@Resource
|
||||||
|
RedisTemplate redisTemplate;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public List<Blogs> findAllBlogs() {
|
public List<Blogs> findAllBlogs() {
|
||||||
return blogsMapper.findAllBlogs();
|
return blogsMapper.findAllBlogs();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
ObjectMapper objectMapper;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取博客的详情
|
||||||
|
*
|
||||||
|
* @param blogId
|
||||||
|
* @return {@link GetBlogDetailVo }
|
||||||
|
*/
|
||||||
@Override
|
@Override
|
||||||
public Blogs findByIdBlogs(int id) {
|
public GetBlogDetailVo findByIdBlogs(int blogId) {
|
||||||
return blogsMapper.findByIdBlogs(id);
|
LambdaQueryWrapper<Blogs> queryWrapper = new LambdaQueryWrapper<>();
|
||||||
|
queryWrapper.eq(Blogs::getId, blogId);
|
||||||
|
Blogs blog = blogsMapper.selectOne(queryWrapper);
|
||||||
|
|
||||||
|
GetBlogDetailVo getBlogDetailVo = new GetBlogDetailVo();
|
||||||
|
getBlogDetailVo.setId(blog.getId());
|
||||||
|
getBlogDetailVo.setTitle(blog.getTitle());
|
||||||
|
getBlogDetailVo.setContent(blog.getContent());
|
||||||
|
getBlogDetailVo.setAuthorId(blog.getAuthorId());
|
||||||
|
getBlogDetailVo.setCreatedTime(blog.getCreatedTime());
|
||||||
|
getBlogDetailVo.setUpdatedTime(blog.getUpdatedTime());
|
||||||
|
// 获取动态计数信息:点赞数、浏览数、收藏数、评论数
|
||||||
|
//TODO 使用 Redis 分布式锁来确保只有一个线程能够从数据库中获取并更新缓存
|
||||||
|
getBlogDetailVo.setLikeCount(getCachedCount(constantConfiguration.BLOG_LIKE_COUNT + blogId, () -> blogsMapper.getLikeCount(blogId)));
|
||||||
|
getBlogDetailVo.setViewCount(getCachedCount(constantConfiguration.BLOG_VIEW_COUNT + blogId, () -> blogsMapper.getViewCount(blogId)));
|
||||||
|
getBlogDetailVo.setCollectCount(getCachedCount(constantConfiguration.BLOG_COLLECT_COUNT + blogId, () -> blogsMapper.getCollectCount(blogId)));
|
||||||
|
getBlogDetailVo.setCommentCount(getCachedCount(constantConfiguration.BLOG_COMMENT_COUNT + blogId, () -> blogsMapper.getCommentCount(blogId)));
|
||||||
|
|
||||||
|
// 获取动态信息的状态:点赞状态,收藏状态
|
||||||
|
getBlogDetailVo.setLikeStatus(getCachedSet(constantConfiguration.BLOG_LIKE_SET + blogId, () -> blogsMapper.getLikeStatus(blogId, RequestHolder.getuserId())));
|
||||||
|
getBlogDetailVo.setCollectStatus(getCachedSet(constantConfiguration.BLOG_COLLECT_SET + blogId, () -> blogsMapper.getCollectStatus(blogId, RequestHolder.getuserId())));
|
||||||
|
// 手动设置额外字段
|
||||||
|
User author = userService.getUser(String.valueOf(blog.getAuthorId()));
|
||||||
|
if (author != null) {
|
||||||
|
getBlogDetailVo.setAuthorName(author.getUsername());
|
||||||
|
getBlogDetailVo.setAvatar(author.getAvatar());
|
||||||
|
}
|
||||||
|
|
||||||
|
return getBlogDetailVo;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 检查是否有该博客的动态计数,若无,则从数据库中查询并更新redis,若有,则直接从redis中获取
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* @param dbQuery
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
private String getCachedCount(String key, Supplier<String> dbQuery) {
|
||||||
|
String countExpiredKey = key + ":expired"; //逻辑过期的key
|
||||||
|
// 尝试从 Redis 中获取计数
|
||||||
|
String count = (String) redisTemplate.opsForValue().get(key);
|
||||||
|
|
||||||
|
// 如果缓存中没有,则从数据库获取并写入缓存
|
||||||
|
if (count == null) {
|
||||||
|
count = dbQuery.get();
|
||||||
|
if (count != null) {
|
||||||
|
redisTemplate.opsForValue().set(key, count.toString());
|
||||||
|
|
||||||
|
// 如果逻辑过期标志不存在,则创建并设置为 23 小时,确保缓存到期前同步
|
||||||
|
if (!Boolean.TRUE.equals(redisTemplate.hasKey(countExpiredKey))) {
|
||||||
|
redisTemplate.opsForValue().set(countExpiredKey, "true", 23, TimeUnit.HOURS);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
count = "0"; // 如果数据库也没有数据,返回默认值 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 检查redis的set集合中是否存在该用户,若无,将当前用户的点赞状态更新至redis
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* @param dbQuery
|
||||||
|
* @return {@link Integer }
|
||||||
|
*/
|
||||||
|
private Integer getCachedSet(String key, Supplier<Integer> dbQuery) {
|
||||||
|
String userId = RequestHolder.getuserId().toString();
|
||||||
|
String blogId = key.split(":")[2];
|
||||||
|
|
||||||
|
// 首先检查 Redis 中是否存在该 key
|
||||||
|
Boolean keyExists = redisTemplate.hasKey(key);
|
||||||
|
if (Boolean.FALSE.equals(keyExists)) {
|
||||||
|
// 如果 Redis 中没有该 key,从数据库中获取点赞状态
|
||||||
|
Integer status = dbQuery.get();
|
||||||
|
|
||||||
|
// 获取该博客用户集合存入redis的set中
|
||||||
|
List<String> userIds = blogsMapper.getUserIdsByBlogId(blogId);
|
||||||
|
if (userIds != null && !userIds.isEmpty()) {
|
||||||
|
redisTemplate.opsForSet().add(key, userIds.toArray());
|
||||||
|
}
|
||||||
|
return status;
|
||||||
|
} else {
|
||||||
|
// 如果 Redis 中存在 key,检查用户是否在 Set 中
|
||||||
|
Boolean exists = redisTemplate.opsForSet().isMember(key, userId);
|
||||||
|
return exists ? 1 : 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void addBlogs(addBlogDto blogDto) {
|
public void addBlogs(AddBlogDto blogDto) {
|
||||||
LocalDateTime now = LocalDateTime.now();
|
LocalDateTime now = LocalDateTime.now();
|
||||||
// 博客内容的插入
|
// 博客内容的插入
|
||||||
Blogs blog = new Blogs();
|
Blogs blog = new Blogs();
|
||||||
|
@ -55,4 +177,182 @@ public class BlogsServiceImpl implements BlogService {
|
||||||
public void deleteBlogs(int id) {
|
public void deleteBlogs(int id) {
|
||||||
blogsMapper.deleteBlogs(id);
|
blogsMapper.deleteBlogs(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 根据用户id查询博客
|
||||||
|
*
|
||||||
|
* @param getUserBlogsDto
|
||||||
|
* @return {@link List }<{@link Blogs }>
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Page<Blogs> findBlogsByUserId(GetUserBlogsDto getUserBlogsDto) {
|
||||||
|
User user = userService.getUser(getUserBlogsDto.getUserId());
|
||||||
|
Page<Blogs> pageRequest = new Page<>(getUserBlogsDto.getPage(), getUserBlogsDto.getPageSize());
|
||||||
|
|
||||||
|
LambdaQueryWrapper<Blogs> queryWrapper = new LambdaQueryWrapper<>();
|
||||||
|
queryWrapper.eq(Blogs::getAuthorId, user.getId())
|
||||||
|
.orderByDesc(Blogs::getCreatedTime);
|
||||||
|
return blogsMapper.selectPage(pageRequest, queryWrapper);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 浏览博客增加博客阅读数,查看redis中是否有该key,若无则从数据库中获取
|
||||||
|
*
|
||||||
|
* @param blogId
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void viewBlog(int blogId) {
|
||||||
|
String userId = RequestHolder.getuserId().toString(); // 获取当前用户 ID
|
||||||
|
|
||||||
|
// 定义 Redis 键
|
||||||
|
String viewUserKey = constantConfiguration.BLOG_VIEW_SET + blogId; // 存放访问用户的 Set,带1小时过期
|
||||||
|
String viewCountKey = constantConfiguration.BLOG_VIEW_COUNT + blogId; // 存放浏览数,带7天过期
|
||||||
|
String viewCountExpireKey = constantConfiguration.BLOG_VIEW_COUNT + blogId + ":expired";// 逻辑过期标志
|
||||||
|
|
||||||
|
// 检查用户是否在 1 小时内访问过该博客
|
||||||
|
Boolean isFirstView = redisTemplate.opsForSet().isMember(viewUserKey, userId);
|
||||||
|
|
||||||
|
if (isFirstView == null || !isFirstView) {
|
||||||
|
// 如果用户未访问或 Redis key 不存在,增加浏览数
|
||||||
|
redisTemplate.opsForSet().add(viewUserKey, userId);
|
||||||
|
redisTemplate.expire(viewUserKey, 1, TimeUnit.HOURS);
|
||||||
|
|
||||||
|
// 检查浏览数 key 是否存在
|
||||||
|
String currentViewCount = (String) redisTemplate.opsForValue().get(viewCountKey);
|
||||||
|
if (currentViewCount == null) {
|
||||||
|
// 如果 Redis 中浏览数 key 不存在,从数据库获取并缓存
|
||||||
|
currentViewCount = blogsMapper.getViewCount(blogId);
|
||||||
|
redisTemplate.opsForValue().set(viewCountKey, currentViewCount);
|
||||||
|
|
||||||
|
// 设置逻辑过期标志,确保缓存到期前会被同步到数据库
|
||||||
|
if (!Boolean.TRUE.equals(redisTemplate.hasKey(viewCountExpireKey))) {
|
||||||
|
redisTemplate.opsForValue().set(viewCountExpireKey, "true", 23, TimeUnit.HOURS);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
redisTemplate.opsForValue().increment(viewCountKey);
|
||||||
|
// 如果逻辑过期标志尚未存在,则创建并设置过期时间
|
||||||
|
if (!Boolean.TRUE.equals(redisTemplate.hasKey(viewCountExpireKey))) {
|
||||||
|
redisTemplate.opsForValue().set(viewCountExpireKey, "true", 23, TimeUnit.HOURS);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 点赞博客增加博客点赞数
|
||||||
|
*
|
||||||
|
* @param blogId
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void likeBlog(int blogId) {
|
||||||
|
String userId = RequestHolder.getuserId().toString(); // 获取用户ID
|
||||||
|
|
||||||
|
if (userId == null) {
|
||||||
|
throw new IllegalStateException("用户未登录");
|
||||||
|
}
|
||||||
|
String likeUserKey = constantConfiguration.BLOG_LIKE_SET + blogId; // 用于存放点赞用户的 Set
|
||||||
|
|
||||||
|
String likeCountKey = constantConfiguration.BLOG_LIKE_COUNT + blogId; // 用于存放点赞数
|
||||||
|
String likeCountExpireKey = constantConfiguration.BLOG_LIKE_COUNT + blogId + ":expired"; // 逻辑过期标志
|
||||||
|
String likeUserHashKey = constantConfiguration.BLOG_LIKE_HASH + blogId;// 用于存放点赞用户信息的 Hash
|
||||||
|
String likeUserHashExpireKey = constantConfiguration.BLOG_LIKE_HASH + blogId + ":expired";
|
||||||
|
|
||||||
|
// 检查是否已点赞,并确保没有发生空指针异常
|
||||||
|
Boolean isFirstLike = redisTemplate.opsForSet().isMember(likeUserKey, userId);
|
||||||
|
|
||||||
|
if (isFirstLike == null || !isFirstLike) {
|
||||||
|
redisTemplate.opsForSet().add(likeUserKey, userId);
|
||||||
|
redisTemplate.opsForValue().increment(likeCountKey);
|
||||||
|
redisTemplate.opsForHash().put(likeUserHashKey, userId, "true");
|
||||||
|
|
||||||
|
// 设置过期时间仅在首次操作集合时
|
||||||
|
if (!Boolean.TRUE.equals(redisTemplate.hasKey(likeCountExpireKey))) {
|
||||||
|
redisTemplate.opsForValue().set(likeCountExpireKey, "true", 25, TimeUnit.HOURS);
|
||||||
|
}
|
||||||
|
if (!Boolean.TRUE.equals(redisTemplate.hasKey(likeUserHashExpireKey))) {
|
||||||
|
redisTemplate.opsForValue().set(likeUserHashExpireKey, "true", 25, TimeUnit.HOURS);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
redisTemplate.opsForSet().remove(likeUserKey, userId);
|
||||||
|
redisTemplate.opsForValue().decrement(likeCountKey);
|
||||||
|
redisTemplate.opsForHash().put(likeUserHashKey, userId, "false");
|
||||||
|
|
||||||
|
// 设置逻辑过期标志,以确保下次同步到数据库
|
||||||
|
if (!Boolean.TRUE.equals(redisTemplate.hasKey(likeCountExpireKey))) {
|
||||||
|
redisTemplate.opsForValue().set(likeCountExpireKey, "true", 25, TimeUnit.HOURS);
|
||||||
|
}
|
||||||
|
if (!Boolean.TRUE.equals(redisTemplate.hasKey(likeUserHashExpireKey))) {
|
||||||
|
redisTemplate.opsForValue().set(likeUserHashExpireKey, "true", 25, TimeUnit.HOURS);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 收藏博客 增加博客收藏数 用户收藏夹 博客收藏中间表
|
||||||
|
*
|
||||||
|
* @param collectBlogDto
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void collectBlog(CollectBlogDto collectBlogDto) {
|
||||||
|
String userId = RequestHolder.getuserId().toString(); // 获取用户ID
|
||||||
|
String blogId = collectBlogDto.getBlogId().toString();
|
||||||
|
Integer favoritesId = collectBlogDto.getFavoriteId();
|
||||||
|
if (userId == null) {
|
||||||
|
throw new IllegalStateException("用户未登录");
|
||||||
|
}
|
||||||
|
String collectLockKey = constantConfiguration.BLOG_COLLECT_LOCK + blogId;
|
||||||
|
// 尝试获取锁,设置锁的过期时间为10秒,防止死锁
|
||||||
|
Boolean lockAcquired = redisTemplate.opsForValue().setIfAbsent(collectLockKey, "LOCKED", 10, TimeUnit.SECONDS);
|
||||||
|
|
||||||
|
if (Boolean.TRUE.equals(lockAcquired)) {
|
||||||
|
try {
|
||||||
|
String collectUserKey = constantConfiguration.BLOG_COLLECT_SET + blogId; // 用于存放收藏用户的 Set
|
||||||
|
String collectCountKey = constantConfiguration.BLOG_COLLECT_COUNT + blogId; // 用于存放收藏数
|
||||||
|
String collectCountExpireKey = constantConfiguration.BLOG_COLLECT_COUNT + blogId + ":expired"; // 用于存放收藏数
|
||||||
|
String blogCollectBlogsHashKey = constantConfiguration.BLOG_COLLECT_HASH + favoritesId;
|
||||||
|
String blogCollectBlogsHashExpiredKey = constantConfiguration.BLOG_COLLECT_HASH + favoritesId + ":expired";
|
||||||
|
|
||||||
|
// 检查是否已收藏,并确保没有发生空指针异常
|
||||||
|
Boolean isFirstCollect = redisTemplate.opsForSet().isMember(collectUserKey, userId);
|
||||||
|
// Blogs blog = blogsMapper.selectById(collectBlogDto.getBlogId());
|
||||||
|
// String blogJson = objectMapper.writeValueAsString(blog);
|
||||||
|
|
||||||
|
if (isFirstCollect == null || !isFirstCollect) {
|
||||||
|
// 收藏
|
||||||
|
redisTemplate.opsForSet().add(collectUserKey, userId);
|
||||||
|
redisTemplate.opsForValue().increment(collectCountKey);
|
||||||
|
|
||||||
|
// 将收藏的博客缓存到用户收藏夹的博客列表中
|
||||||
|
redisTemplate.opsForHash().put(blogCollectBlogsHashKey, blogId, "true");
|
||||||
|
|
||||||
|
// 设置逻辑过期时间 确保能同步到数据中
|
||||||
|
if (!Boolean.TRUE.equals(redisTemplate.hasKey(collectCountExpireKey))) {
|
||||||
|
redisTemplate.opsForValue().set(collectCountExpireKey, "true", 23, TimeUnit.HOURS);
|
||||||
|
}
|
||||||
|
if (!Boolean.TRUE.equals(redisTemplate.hasKey(blogCollectBlogsHashExpiredKey))) {
|
||||||
|
redisTemplate.opsForValue().set(blogCollectBlogsHashExpiredKey, "true", 23, TimeUnit.HOURS);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// 取消收藏
|
||||||
|
redisTemplate.opsForSet().remove(collectUserKey, userId);
|
||||||
|
redisTemplate.opsForValue().decrement(collectCountKey);
|
||||||
|
redisTemplate.opsForHash().put(blogCollectBlogsHashKey, blogId, "false");
|
||||||
|
|
||||||
|
// 设置逻辑过期时间 确保能同步到数据中
|
||||||
|
if (!Boolean.TRUE.equals(redisTemplate.hasKey(collectCountExpireKey))) {
|
||||||
|
redisTemplate.opsForValue().set(collectCountExpireKey, "true", 23, TimeUnit.HOURS);
|
||||||
|
}
|
||||||
|
if (!Boolean.TRUE.equals(redisTemplate.hasKey(blogCollectBlogsHashExpiredKey))) {
|
||||||
|
redisTemplate.opsForValue().set(blogCollectBlogsHashExpiredKey, "true", 23, TimeUnit.HOURS);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
// 释放锁
|
||||||
|
redisTemplate.delete(collectLockKey);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.warn("无法获取分布式锁,用户 {} 收藏博客 {} 操作被跳过", userId, blogId);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,6 +18,8 @@ import java.util.List;
|
||||||
public class CommentServiceImpl implements CommentService {
|
public class CommentServiceImpl implements CommentService {
|
||||||
@Resource
|
@Resource
|
||||||
CommentMapper commentMapper;
|
CommentMapper commentMapper;
|
||||||
|
@Resource
|
||||||
|
UserService userService;
|
||||||
@Override
|
@Override
|
||||||
public List<CommentVo> findAllComment(int article_id) {
|
public List<CommentVo> findAllComment(int article_id) {
|
||||||
return commentMapper.findAllComment(article_id);
|
return commentMapper.findAllComment(article_id);
|
||||||
|
@ -28,7 +30,7 @@ public class CommentServiceImpl implements CommentService {
|
||||||
Comment comment1 = new Comment();
|
Comment comment1 = new Comment();
|
||||||
//对象属性拷贝
|
//对象属性拷贝
|
||||||
BeanUtils.copyProperties(comment,comment1);
|
BeanUtils.copyProperties(comment,comment1);
|
||||||
comment1.setUsername(RequestHolder.getuserId());
|
comment1.setUsername(userService.getUser("").getUsername());
|
||||||
System.out.println(comment1);
|
System.out.println(comment1);
|
||||||
commentMapper.addComment(comment1);
|
commentMapper.addComment(comment1);
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@ import com.xubx.springboot_01demo.mapper.RelationshipMapper;
|
||||||
import com.xubx.springboot_01demo.pojo.Relationship;
|
import com.xubx.springboot_01demo.pojo.Relationship;
|
||||||
import com.xubx.springboot_01demo.pojo.User;
|
import com.xubx.springboot_01demo.pojo.User;
|
||||||
import com.xubx.springboot_01demo.service.RelationshipService;
|
import com.xubx.springboot_01demo.service.RelationshipService;
|
||||||
|
import com.xubx.springboot_01demo.service.UserService;
|
||||||
import com.xubx.springboot_01demo.utils.token.RequestHolder;
|
import com.xubx.springboot_01demo.utils.token.RequestHolder;
|
||||||
import com.xubx.springboot_01demo.vo.UserListVo;
|
import com.xubx.springboot_01demo.vo.UserListVo;
|
||||||
import com.xubx.springboot_01demo.vo.UserVo;
|
import com.xubx.springboot_01demo.vo.UserVo;
|
||||||
|
@ -20,6 +21,8 @@ public class RelationshipServiceImpl implements RelationshipService {
|
||||||
|
|
||||||
@Resource
|
@Resource
|
||||||
RelationshipMapper relationshipMapper;
|
RelationshipMapper relationshipMapper;
|
||||||
|
@Resource
|
||||||
|
UserService userService;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -31,8 +34,9 @@ public class RelationshipServiceImpl implements RelationshipService {
|
||||||
@Override
|
@Override
|
||||||
public List<UserVo> queryUser(String username) {
|
public List<UserVo> queryUser(String username) {
|
||||||
List<UserVo> userDtos = new ArrayList<>();
|
List<UserVo> userDtos = new ArrayList<>();
|
||||||
|
|
||||||
//将User转换为UserDto
|
//将User转换为UserDto
|
||||||
for(User u : relationshipMapper.findUserByUsername(RequestHolder.getuserId(),username)){
|
for(User u : relationshipMapper.findUserByUsername(userService.getUser("").getUsername(),username)){
|
||||||
UserVo userDto = new UserVo();
|
UserVo userDto = new UserVo();
|
||||||
//将User的属性复制到UserDto
|
//将User的属性复制到UserDto
|
||||||
BeanUtils.copyProperties(u,userDto);
|
BeanUtils.copyProperties(u,userDto);
|
||||||
|
|
|
@ -3,17 +3,38 @@ package com.xubx.springboot_01demo.service.impl;
|
||||||
|
|
||||||
import com.alibaba.fastjson.JSONArray;
|
import com.alibaba.fastjson.JSONArray;
|
||||||
import com.alibaba.fastjson.JSONObject;
|
import com.alibaba.fastjson.JSONObject;
|
||||||
|
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
||||||
|
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.dto.user.EditMaterialDto;
|
||||||
|
import com.xubx.springboot_01demo.dto.user.NewFavoritesDto;
|
||||||
|
import com.xubx.springboot_01demo.mapper.BlogsMapper;
|
||||||
import com.xubx.springboot_01demo.mapper.RelationshipMapper;
|
import com.xubx.springboot_01demo.mapper.RelationshipMapper;
|
||||||
|
import com.xubx.springboot_01demo.mapper.UserFavoriteMapper;
|
||||||
import com.xubx.springboot_01demo.mapper.UserMapper;
|
import com.xubx.springboot_01demo.mapper.UserMapper;
|
||||||
|
import com.xubx.springboot_01demo.pojo.Blogs;
|
||||||
import com.xubx.springboot_01demo.pojo.Relationship;
|
import com.xubx.springboot_01demo.pojo.Relationship;
|
||||||
import com.xubx.springboot_01demo.pojo.User;
|
import com.xubx.springboot_01demo.pojo.User;
|
||||||
|
import com.xubx.springboot_01demo.pojo.UserFavorite;
|
||||||
import com.xubx.springboot_01demo.service.UserService;
|
import com.xubx.springboot_01demo.service.UserService;
|
||||||
import com.xubx.springboot_01demo.utils.api.Result;
|
import com.xubx.springboot_01demo.utils.token.RequestHolder;
|
||||||
|
import com.xubx.springboot_01demo.utils.token.TokenGenerate;
|
||||||
|
import com.xubx.springboot_01demo.vo.getUserInfoVo;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.data.redis.core.RedisTemplate;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
import javax.annotation.Resource;
|
import javax.annotation.Resource;
|
||||||
|
import javax.servlet.http.HttpSession;
|
||||||
|
import java.sql.Timestamp;
|
||||||
|
import java.time.LocalDateTime;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
//Service进行具体的业务处理
|
//Service进行具体的业务处理
|
||||||
@Service
|
@Service
|
||||||
|
@ -22,28 +43,45 @@ public class UserServiceImpl implements UserService {
|
||||||
UserMapper userMapper;
|
UserMapper userMapper;
|
||||||
@Resource
|
@Resource
|
||||||
RelationshipMapper relationshipMapper;
|
RelationshipMapper relationshipMapper;
|
||||||
|
@Resource
|
||||||
|
BlogsMapper blogMapper;
|
||||||
|
@Resource
|
||||||
|
UserFavoriteMapper userFavoriteMapper;
|
||||||
|
@Autowired
|
||||||
|
RedisTemplate redisTemplate;
|
||||||
|
@Resource
|
||||||
|
private HttpSession session;
|
||||||
|
@Autowired
|
||||||
|
ObjectMapper objectMapper;
|
||||||
|
|
||||||
//登陆,获取User对象
|
//登陆,获取User对象
|
||||||
@Override
|
public String login(User user) {
|
||||||
public Boolean findUserByUsername(User user) {
|
LambdaQueryWrapper<User> queryWrapper = new LambdaQueryWrapper<>();
|
||||||
try {
|
queryWrapper.eq(User::getUsername, user.getUsername());
|
||||||
User userByUsername = userMapper.findUserByUsername(user.getUsername());
|
User userByUsername = userMapper.selectOne(queryWrapper);
|
||||||
|
RequestHolder.add(userByUsername.getId());
|
||||||
|
|
||||||
if (Objects.equals(userByUsername.getPassword(), user.getPassword())) {
|
// 校验用户是否存在以及密码是否匹配
|
||||||
return true;
|
if (userByUsername != null && Objects.equals(userByUsername.getPassword(), user.getPassword())) {
|
||||||
}
|
// 生成 Token
|
||||||
return false;
|
String token = new TokenGenerate().generateToken(userByUsername.getId());
|
||||||
} catch (Exception e) {
|
// 将用户名存入 session 中
|
||||||
System.out.println(e);
|
session.setAttribute("username", userByUsername.getUsername());
|
||||||
return false;
|
return token;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 登录失败返回 null
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
//注册,新增
|
//注册,新增
|
||||||
@Override
|
@Override
|
||||||
public Boolean insertUser(User user) {
|
public Boolean insertUser(User user) {
|
||||||
|
LambdaQueryWrapper<User> queryWrapper = new LambdaQueryWrapper<>();
|
||||||
|
queryWrapper.eq(User::getUsername, user.getUsername());
|
||||||
|
|
||||||
// 检查用户名是否已存在
|
// 检查用户名是否已存在
|
||||||
if (userMapper.findUserByUsername(user.getUsername()) != null) {
|
if (userMapper.selectOne(queryWrapper) != null) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
// 用户名不存在,插入新用户
|
// 用户名不存在,插入新用户
|
||||||
|
@ -52,8 +90,11 @@ public class UserServiceImpl implements UserService {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void addAvatar(String path, String username) {
|
public void addAvatar(String path, String userId) {
|
||||||
userMapper.addAvatar(path, username);
|
LambdaUpdateWrapper<User> updateWrapper = new LambdaUpdateWrapper<>();
|
||||||
|
updateWrapper.eq(User::getId, userId)
|
||||||
|
.set(User::getAvatar, path);
|
||||||
|
userMapper.update(null, updateWrapper);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -62,13 +103,11 @@ public class UserServiceImpl implements UserService {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean usernameChange(String usernameNow, String username) {
|
public void usernameChange(String username) {
|
||||||
// 检查用户名是否已存在
|
LambdaUpdateWrapper<User> updateWrapper = new LambdaUpdateWrapper<>();
|
||||||
if (userMapper.findUserByUsername(username) != null) {
|
updateWrapper.eq(User::getId, RequestHolder.getuserId())
|
||||||
return false;
|
.set(User::getUsername, username);
|
||||||
}
|
userMapper.update(null, updateWrapper);
|
||||||
userMapper.usernameChange(usernameNow, username);
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -100,4 +139,181 @@ public class UserServiceImpl implements UserService {
|
||||||
result.put("friendRequests", jsonArray);
|
result.put("friendRequests", jsonArray);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取用户
|
||||||
|
*
|
||||||
|
* @param userId
|
||||||
|
* @return {@link User }
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public User getUser(String userId) {
|
||||||
|
User user;
|
||||||
|
if (userId != null && !userId.equals("")) {
|
||||||
|
user = userMapper.selectById(userId);
|
||||||
|
} else {
|
||||||
|
// 如果userId为空,则获取当前登录用户的userId
|
||||||
|
user = userMapper.selectById(RequestHolder.getuserId());
|
||||||
|
}
|
||||||
|
return user;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取用户信息
|
||||||
|
*
|
||||||
|
* @param userId
|
||||||
|
* @return {@link getUserInfoVo }
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public getUserInfoVo getUserInfo(String userId) {
|
||||||
|
User user = getUser(userId);
|
||||||
|
final Integer finalUserId = user.getId();
|
||||||
|
|
||||||
|
getUserInfoVo getUserInfoVo = new getUserInfoVo();
|
||||||
|
getUserInfoVo.setUsername(user.getUsername());
|
||||||
|
getUserInfoVo.setAvatar(user.getAvatar());
|
||||||
|
getUserInfoVo.setBio(user.getBio());
|
||||||
|
getUserInfoVo.setJoinBlogTime(user.getCreatedTime());
|
||||||
|
getUserInfoVo.setTotalViewCount(getCachedValue(constantConfiguration.USER_TOTAL_VIEW_COUNT + finalUserId, () -> blogMapper.getTotalViewCount(finalUserId)));
|
||||||
|
getUserInfoVo.setTotalBlogCount(getCachedValue(constantConfiguration.USER_TOTAL_BLOG_COUNT + finalUserId, () -> blogMapper.getTotalBlogCount(finalUserId)));
|
||||||
|
getUserInfoVo.setTotalCollectCount(getCachedValue(constantConfiguration.USER_TOTAL_COLLECT_COUNT + finalUserId, () -> blogMapper.getTotalCollectCount(finalUserId)));
|
||||||
|
getUserInfoVo.setTotalLikeCount(getCachedValue(constantConfiguration.USER_TOTAL_LIKE_COUNT + finalUserId, () -> blogMapper.getTotalLikeCount(finalUserId)));
|
||||||
|
getUserInfoVo.setTotalCommentCount(getCachedValue(constantConfiguration.USER_TOTAL_COMMENT_COUNT + finalUserId, () -> blogMapper.getTotalCommentCount(finalUserId)));
|
||||||
|
getUserInfoVo.setTotalShareCount(getCachedValue(constantConfiguration.USER_TOTAL_SHARE_COUNT + finalUserId, () -> blogMapper.getTotalShareCount(finalUserId)));
|
||||||
|
getUserInfoVo.setTotalFollowerCount(getCachedValue(constantConfiguration.USER_TOTAL_FOLLOWER_COUNT + finalUserId, () -> relationshipMapper.getTotalFollowerCount(finalUserId)));
|
||||||
|
return getUserInfoVo;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取缓存值通用方法
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* @param dbQuery
|
||||||
|
* @return {@link Integer }
|
||||||
|
*/
|
||||||
|
private String getCachedValue(String key, Supplier<String> dbQuery) {
|
||||||
|
String value = (String) redisTemplate.opsForValue().get(key);
|
||||||
|
if (value == null) {
|
||||||
|
value = dbQuery.get();
|
||||||
|
redisTemplate.opsForValue().set(key, value, 30, TimeUnit.MINUTES);
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 修改个人资料
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void editMaterial(EditMaterialDto editMaterialDto) {
|
||||||
|
LocalDateTime now = LocalDateTime.now();
|
||||||
|
User user = new User();
|
||||||
|
user.setId(RequestHolder.getuserId());
|
||||||
|
user.setUsername(editMaterialDto.getUsername());
|
||||||
|
user.setAvatar(editMaterialDto.getAvatar());
|
||||||
|
user.setGender(editMaterialDto.getGender());
|
||||||
|
user.setBio(editMaterialDto.getBio());
|
||||||
|
user.setUpdateTime(Timestamp.valueOf(now));
|
||||||
|
userMapper.updateById(user);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 新建收藏夹
|
||||||
|
*
|
||||||
|
* @param newFavorites
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String newFavorites(NewFavoritesDto newFavorites) {
|
||||||
|
// 检查标题是否已存在
|
||||||
|
LambdaQueryWrapper<UserFavorite> wrapper = new LambdaQueryWrapper<>();
|
||||||
|
wrapper.eq(UserFavorite::getName, newFavorites.getName())
|
||||||
|
.eq(UserFavorite::getUserId, RequestHolder.getuserId());
|
||||||
|
int count = userFavoriteMapper.selectCount(wrapper);
|
||||||
|
|
||||||
|
if (count > 0) {
|
||||||
|
return "该收藏夹已存在";
|
||||||
|
} else {
|
||||||
|
UserFavorite userFavorite = new UserFavorite();
|
||||||
|
userFavorite.setName(newFavorites.getName());
|
||||||
|
userFavorite.setUserId(RequestHolder.getuserId());
|
||||||
|
userFavorite.setDescription(newFavorites.getDescription());
|
||||||
|
userFavorite.setCreatedTime(Timestamp.valueOf(LocalDateTime.now()));
|
||||||
|
// 检查是否由收藏夹状态
|
||||||
|
if (newFavorites.getFavoriteStatus() != null) {
|
||||||
|
userFavorite.setFavoriteStatus("私密");
|
||||||
|
} else {
|
||||||
|
userFavorite.setFavoriteStatus("公开");
|
||||||
|
}
|
||||||
|
userFavoriteMapper.insert(userFavorite);
|
||||||
|
return null; // 创建成功,无错误消息
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取用户的收藏夹列表
|
||||||
|
*
|
||||||
|
* @param userId
|
||||||
|
* @return {@link List }<{@link UserFavorite }>
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public List<UserFavorite> getUserFavorites(String userId) {
|
||||||
|
Integer currentUserId = RequestHolder.getuserId();
|
||||||
|
|
||||||
|
if (currentUserId == null) {
|
||||||
|
throw new IllegalStateException("用户未登录");
|
||||||
|
}
|
||||||
|
|
||||||
|
LambdaQueryWrapper<UserFavorite> wrapper = new LambdaQueryWrapper<>();
|
||||||
|
// 首先判断是否是获取当前用户的收藏夹列表
|
||||||
|
if (userId == null || userId.isEmpty() || userId.equals(currentUserId.toString())) {
|
||||||
|
// 获取当前用户的全部收藏夹
|
||||||
|
// TODO 收藏夹中的博客数量不是实时的
|
||||||
|
wrapper.eq(UserFavorite::getUserId, currentUserId)
|
||||||
|
.orderByDesc(UserFavorite::getCreatedTime);
|
||||||
|
return userFavoriteMapper.selectList(wrapper);
|
||||||
|
} else {
|
||||||
|
// 获取指定用户的公开的收藏夹
|
||||||
|
wrapper.eq(UserFavorite::getUserId, userId)
|
||||||
|
.eq(UserFavorite::getFavoriteStatus, "公开")
|
||||||
|
.orderByDesc(UserFavorite::getCreatedTime);
|
||||||
|
return userFavoriteMapper.selectList(wrapper);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取收藏夹中的博客列表
|
||||||
|
*
|
||||||
|
* @param favoriteId
|
||||||
|
* @return {@link List }<{@link Blogs }>
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public List<Blogs> getBlogsByFavoriteId(String favoriteId) {
|
||||||
|
Integer userId = RequestHolder.getuserId();
|
||||||
|
if (userId == null) {
|
||||||
|
throw new IllegalStateException("用户未登录");
|
||||||
|
}
|
||||||
|
|
||||||
|
// 首先获取数据库中的数据
|
||||||
|
List<Blogs> blogsFromDb = blogMapper.getBlogsByFavoriteId(favoriteId);
|
||||||
|
|
||||||
|
// 然后获取缓存中的数据
|
||||||
|
String userCollectBlogsHashKey = constantConfiguration.BLOG_COLLECT_HASH + favoriteId;
|
||||||
|
if (redisTemplate.hasKey(userCollectBlogsHashKey)) {
|
||||||
|
// 缓存中有数据,需要拼接到数据库获取到的数据
|
||||||
|
Set<String> blogSet = redisTemplate.opsForHash().keys(userCollectBlogsHashKey);
|
||||||
|
if (blogSet != null && !blogSet.isEmpty()) {
|
||||||
|
for (String blogIdStr : blogSet) {
|
||||||
|
String collectStatus = (String) redisTemplate.opsForHash().get(userCollectBlogsHashKey, blogIdStr);
|
||||||
|
Blogs blog = blogMapper.findByIdBlogs(Integer.parseInt(blogIdStr));
|
||||||
|
if ("true".equals(collectStatus)) {
|
||||||
|
blogsFromDb.add(blog);
|
||||||
|
} else if ("false".equals(collectStatus)) {
|
||||||
|
blogsFromDb.remove(blog);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return blogsFromDb;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,13 +4,13 @@ package com.xubx.springboot_01demo.utils.token;
|
||||||
* 用于存储当前用户的id
|
* 用于存储当前用户的id
|
||||||
*/
|
*/
|
||||||
public class RequestHolder {
|
public class RequestHolder {
|
||||||
private static final ThreadLocal<String> userHolder = new ThreadLocal<>();
|
private static final ThreadLocal<Integer> userHolder = new ThreadLocal<>();
|
||||||
|
|
||||||
public static void add(String userId) {
|
public static void add(Integer userId) {
|
||||||
userHolder.set(userId);
|
userHolder.set(userId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static String getuserId() {
|
public static Integer getuserId() {
|
||||||
return userHolder.get();
|
return userHolder.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -12,13 +12,13 @@ public class TokenGenerate {
|
||||||
private static final long EXPIRE_TIME = 24 * 60 * 60 * 1000;
|
private static final long EXPIRE_TIME = 24 * 60 * 60 * 1000;
|
||||||
private static final String TOKEN_SECRET = "tokenqkj"; //密钥盐
|
private static final String TOKEN_SECRET = "tokenqkj"; //密钥盐
|
||||||
|
|
||||||
public String generateToken(String username) {
|
public String generateToken(int userId) {
|
||||||
String token = null;
|
String token = null;
|
||||||
try {
|
try {
|
||||||
Date expiresAt = new Date(System.currentTimeMillis() + EXPIRE_TIME);
|
Date expiresAt = new Date(System.currentTimeMillis() + EXPIRE_TIME);
|
||||||
token = JWT.create()
|
token = JWT.create()
|
||||||
.withIssuer("auth0")
|
.withIssuer("auth0")
|
||||||
.withClaim("username", username)
|
.withClaim("userId", userId)
|
||||||
.withExpiresAt(expiresAt)
|
.withExpiresAt(expiresAt)
|
||||||
.sign(Algorithm.HMAC256(TOKEN_SECRET));
|
.sign(Algorithm.HMAC256(TOKEN_SECRET));
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
|
@ -38,10 +38,10 @@ public class TokenGenerate {
|
||||||
}
|
}
|
||||||
JWTVerifier verifier = JWT.require(Algorithm.HMAC256(TOKEN_SECRET)).withIssuer("auth0").build();
|
JWTVerifier verifier = JWT.require(Algorithm.HMAC256(TOKEN_SECRET)).withIssuer("auth0").build();
|
||||||
DecodedJWT jwt = verifier.verify(token);
|
DecodedJWT jwt = verifier.verify(token);
|
||||||
RequestHolder.add(jwt.getClaim("username").asString());
|
RequestHolder.add(jwt.getClaim("userId").asInt());
|
||||||
System.out.println("认证通过:");
|
System.out.println("认证通过:");
|
||||||
System.out.println("issuer: " + jwt.getIssuer());
|
System.out.println("issuer: " + jwt.getIssuer());
|
||||||
System.out.println("username: " + jwt.getClaim("username").asString());
|
System.out.println("userId: " + jwt.getClaim("userId").asString());
|
||||||
System.out.println("过期时间: " + jwt.getExpiresAt());
|
System.out.println("过期时间: " + jwt.getExpiresAt());
|
||||||
return true;
|
return true;
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
|
|
|
@ -0,0 +1,54 @@
|
||||||
|
package com.xubx.springboot_01demo.vo;
|
||||||
|
|
||||||
|
import io.swagger.annotations.ApiModel;
|
||||||
|
import io.swagger.annotations.ApiModelProperty;
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
import java.sql.Timestamp;
|
||||||
|
|
||||||
|
@Data
|
||||||
|
@ApiModel(value = "getBlogDetail", description = "获取博客详情")
|
||||||
|
public class GetBlogDetailVo {
|
||||||
|
private int id;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "标题")
|
||||||
|
private String title;
|
||||||
|
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "内容")
|
||||||
|
private String content;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "作者ID")
|
||||||
|
private Integer authorId;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "作者名")
|
||||||
|
private String authorName;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "作者头像")
|
||||||
|
private String avatar;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "点赞数")
|
||||||
|
private String likeCount;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "点赞状态: 0未点赞 1已点赞")
|
||||||
|
private int likeStatus;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "浏览数")
|
||||||
|
private String viewCount;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "评论数")
|
||||||
|
private String commentCount;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "收藏数")
|
||||||
|
private String collectCount;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "收藏状态:0未收藏 1已收藏")
|
||||||
|
private int collectStatus;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "更新时间")
|
||||||
|
private Timestamp updatedTime;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "创建时间")
|
||||||
|
private Timestamp createdTime;
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,44 @@
|
||||||
|
package com.xubx.springboot_01demo.vo;
|
||||||
|
|
||||||
|
import io.swagger.annotations.ApiModel;
|
||||||
|
import io.swagger.annotations.ApiModelProperty;
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
import java.sql.Timestamp;
|
||||||
|
|
||||||
|
@Data
|
||||||
|
@ApiModel(description = "获取当前用户信息出参")
|
||||||
|
public class getUserInfoVo {
|
||||||
|
@ApiModelProperty(value = "用户名")
|
||||||
|
private String username;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "头像")
|
||||||
|
private String avatar;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "个人简介")
|
||||||
|
private String bio;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "总访问量")
|
||||||
|
private String totalViewCount;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "总博客数")
|
||||||
|
private String totalBlogCount;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "粉丝数量")
|
||||||
|
private String totalFollowerCount;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "总收藏数")
|
||||||
|
private String totalCollectCount;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "加入时间")
|
||||||
|
private Timestamp joinBlogTime;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "总点赞数")
|
||||||
|
private String totalLikeCount;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "总评论数")
|
||||||
|
private String totalCommentCount;
|
||||||
|
|
||||||
|
@ApiModelProperty(value = "总分享数")
|
||||||
|
private String totalShareCount;
|
||||||
|
}
|
|
@ -11,6 +11,7 @@ spring:
|
||||||
host: 62.234.217.137
|
host: 62.234.217.137
|
||||||
port: 6379
|
port: 6379
|
||||||
password: LSHCwjr6ZN4hzCxS
|
password: LSHCwjr6ZN4hzCxS
|
||||||
|
database: 0
|
||||||
web:
|
web:
|
||||||
resources:
|
resources:
|
||||||
static-locations: classpath:/META-INF/resources/,classpath:/resources/,classpath:/static/,classpath:/public/
|
static-locations: classpath:/META-INF/resources/,classpath:/resources/,classpath:/static/,classpath:/public/
|
||||||
|
@ -27,4 +28,5 @@ logging:
|
||||||
qiniu:
|
qiniu:
|
||||||
accessKey: McJYQOg3S6PhkljCg2DQeggBQTcGcUZyK_s6vkxy
|
accessKey: McJYQOg3S6PhkljCg2DQeggBQTcGcUZyK_s6vkxy
|
||||||
secretKey: 5hUuHs-BEzJIEKPCEj292h14evv5L59bDSZuYXa-
|
secretKey: 5hUuHs-BEzJIEKPCEj292h14evv5L59bDSZuYXa-
|
||||||
bucket: xubx-blog
|
bucket: xubx-blog
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,6 @@
|
||||||
|
<?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.xubx.springboot_01demo.mapper.BlogLikeMapper">
|
||||||
|
|
||||||
|
</mapper>
|
|
@ -19,6 +19,51 @@
|
||||||
<select id="findByIdBlogs" resultMap="BaseResultMap">
|
<select id="findByIdBlogs" resultMap="BaseResultMap">
|
||||||
select * from blog where id = #{id}
|
select * from blog where id = #{id}
|
||||||
</select>
|
</select>
|
||||||
|
<select id="getTotalViewCount" resultType="java.lang.String">
|
||||||
|
select sum(view_count) from blog where author_id = #{userId}
|
||||||
|
</select>
|
||||||
|
<select id="getTotalBlogCount" resultType="java.lang.String">
|
||||||
|
select count(*) from blog where author_id = #{userId}
|
||||||
|
</select>
|
||||||
|
<select id="getTotalCollectCount" resultType="java.lang.String">
|
||||||
|
select sum(collect_count) from blog where author_id = #{userId}
|
||||||
|
</select>
|
||||||
|
<select id="getTotalLikeCount" resultType="java.lang.String">
|
||||||
|
select sum(like_count) from blog where author_id = #{userId}
|
||||||
|
</select>
|
||||||
|
<select id="getTotalCommentCount" resultType="java.lang.String">
|
||||||
|
select sum(comment_count) from blog where author_id = #{userId}
|
||||||
|
</select>
|
||||||
|
<select id="getTotalShareCount" resultType="java.lang.String">
|
||||||
|
select sum(share_count) from blog where author_id = #{userId}
|
||||||
|
</select>
|
||||||
|
<select id="getLikeStatus" resultType="java.lang.Integer">
|
||||||
|
select count(*) from blog_like where user_id = #{userId} and blog_id = #{blogId}
|
||||||
|
</select>
|
||||||
|
<select id="getCollectStatus" resultType="java.lang.Integer">
|
||||||
|
SELECT count(*)
|
||||||
|
FROM user_favorite_blog t1
|
||||||
|
JOIN (SELECT id FROM user_favorite WHERE user_id = #{userId}) t2 ON t1.favorite_id = t2.id
|
||||||
|
WHERE t1.blog_id = #{blogId}
|
||||||
|
</select>
|
||||||
|
<select id="getLikeCount" resultType="java.lang.String">
|
||||||
|
select like_count from blog where id = #{blogId}
|
||||||
|
</select>
|
||||||
|
<select id="getViewCount" resultType="java.lang.String">
|
||||||
|
select view_count from blog where id = #{blogId}
|
||||||
|
</select>
|
||||||
|
<select id="getCollectCount" resultType="java.lang.String">
|
||||||
|
select collect_count from blog where id = #{blogId}
|
||||||
|
</select>
|
||||||
|
<select id="getCommentCount" resultType="java.lang.String">
|
||||||
|
select comment_count from blog where id = #{blogId}
|
||||||
|
</select>
|
||||||
|
<select id="getBlogsByFavoriteId" resultType="com.xubx.springboot_01demo.pojo.Blogs">
|
||||||
|
select * from blog where id in (select blog_id from user_favorite_blog where favorite_id = #{favoriteId})
|
||||||
|
</select>
|
||||||
|
<select id="getUserIdsByBlogId" resultType="java.lang.String">
|
||||||
|
SELECT user_id FROM blog_like WHERE blog_id = #{blogId}
|
||||||
|
</select>
|
||||||
|
|
||||||
<!--新增博客-->
|
<!--新增博客-->
|
||||||
<insert id="addBlogs" parameterType="com.xubx.springboot_01demo.pojo.Blogs">
|
<insert id="addBlogs" parameterType="com.xubx.springboot_01demo.pojo.Blogs">
|
||||||
|
@ -29,6 +74,10 @@
|
||||||
insert into blog_category(blog_id, category_id)
|
insert into blog_category(blog_id, category_id)
|
||||||
values (#{blogId}, #{categoryId})
|
values (#{blogId}, #{categoryId})
|
||||||
</insert>
|
</insert>
|
||||||
|
<insert id="collectBlog">
|
||||||
|
insert into user_favorite_blog(blog_id, favorite_id)
|
||||||
|
values (#{blogId}, #{favoriteId})
|
||||||
|
</insert>
|
||||||
|
|
||||||
<!--更新博客-->
|
<!--更新博客-->
|
||||||
<update id="updateBlogs" parameterType="com.xubx.springboot_01demo.pojo.Blogs">
|
<update id="updateBlogs" parameterType="com.xubx.springboot_01demo.pojo.Blogs">
|
||||||
|
|
|
@ -57,4 +57,7 @@
|
||||||
and status = 0
|
and status = 0
|
||||||
and initiator != #{username}
|
and initiator != #{username}
|
||||||
</select>
|
</select>
|
||||||
|
<select id="getTotalFollowerCount" resultType="java.lang.String">
|
||||||
|
select count(*) from relationship where friend = #{userId} and status = 1
|
||||||
|
</select>
|
||||||
</mapper>
|
</mapper>
|
|
@ -0,0 +1,6 @@
|
||||||
|
<?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.xubx.springboot_01demo.mapper.UserFavoriteBlogMapper">
|
||||||
|
|
||||||
|
</mapper>
|
|
@ -0,0 +1,6 @@
|
||||||
|
<?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.xubx.springboot_01demo.mapper.UserFavoriteMapper">
|
||||||
|
|
||||||
|
</mapper>
|
|
@ -9,10 +9,6 @@
|
||||||
<result column="password" property="password" />
|
<result column="password" property="password" />
|
||||||
<result column="avatar" property="avatar"/>
|
<result column="avatar" property="avatar"/>
|
||||||
</resultMap>
|
</resultMap>
|
||||||
<!-- 查询用户 -->
|
|
||||||
<select id="findUserByUsername" resultMap="BaseResultMap">
|
|
||||||
select username, password from user where username = #{username}
|
|
||||||
</select>
|
|
||||||
<!-- 插入用户 -->
|
<!-- 插入用户 -->
|
||||||
<insert id="insertUser" parameterType="com.xubx.springboot_01demo.pojo.User" useGeneratedKeys="true" keyProperty="id">
|
<insert id="insertUser" parameterType="com.xubx.springboot_01demo.pojo.User" useGeneratedKeys="true" keyProperty="id">
|
||||||
insert into user(username, password) values(#{username}, #{password})
|
insert into user(username, password) values(#{username}, #{password})
|
||||||
|
|
Loading…
Reference in New Issue