1. 发布探店笔记
1.1 需求分析
这里涉及两个接口,一个是上传照片的接口,一个是发布博客的接口,接下来我们实现这两个接口。
1.2 代码实现
上传图片代码
@Slf4j
@RestController
@RequestMapping("upload")
public class UploadController {
@PostMapping("blog")
public Result uploadImage(@RequestParam("file") MultipartFile image) {
try {
// 获取原始文件名称
String originalFilename = image.getOriginalFilename();
// 生成新文件名
String fileName = createNewFileName(originalFilename);
// 保存文件
image.transferTo(new File(SystemConstants.IMAGE_UPLOAD_DIR, fileName));
// 返回结果
log.debug("文件上传成功,{}", fileName);
return Result.ok(fileName);
} catch (IOException e) {
throw new RuntimeException("文件上传失败", e);
}
}
@GetMapping("/blog/delete")
public Result deleteBlogImg(@RequestParam("name") String filename) {
File file = new File(SystemConstants.IMAGE_UPLOAD_DIR, filename);
if (file.isDirectory()) {
return Result.fail("错误的文件名称");
}
FileUtil.del(file);
return Result.ok();
}
private String createNewFileName(String originalFilename) {
// 获取后缀
String suffix = StrUtil.subAfter(originalFilename, ".", true);
// 生成目录
String name = UUID.randomUUID().toString();
int hash = name.hashCode();
int d1 = hash & 0xF;
int d2 = (hash >> 4) & 0xF;
// 判断目录是否存在
File dir = new File(SystemConstants.IMAGE_UPLOAD_DIR, StrUtil.format("/blogs/{}/{}", d1, d2));
if (!dir.exists()) {
dir.mkdirs();
}
// 生成文件名
return StrUtil.format("/blogs/{}/{}/{}.{}", d1, d2, name, suffix);
}
}
发布笔记功能
@Override
public Result saveBlog(Blog blog) {
//1. 获取当前用户
UserDTO user = UserHolder.getUser();
//2. 新增博客信息
Long userId = user.getId();
blog.setUserId(userId);
boolean success = this.save(blog);
//3. 添加不成功就不推送
if(!success){
return Result.fail("添加博客失败");
}
//4. 返回博客id
return Result.ok(blog.getId());
}
1.3 添加查看笔记接口
在程序主界面,我们可以浏览到用户发布的探店笔记,当我们点击探店笔记时,页面会发起一个请求,获取笔记相关数据进行展示,接下来我们完成一下这个接口。
@Override
public Result queryBlogById(Long id) {
Blog blog = this.getById(id);
if (blog == null) {
return Result.fail("笔记不存在");
}
queryBlogUser(blog);
return Result.ok(blog);
}
//获取博客的用户信息
private void queryBlogUser(Blog blog) {
Long userId = blog.getUserId();
User user = userService.getById(userId);
blog.setName(user.getNickName());
blog.setIcon(user.getIcon());
}
2. 点赞功能
2.1 需求分析
2.2 代码实现
我们分析一下如何实现点赞功能:
首先,一个用户只能点赞一次,那我们如何判断用户是否点赞过该笔记呢?想想,我们是不是能将点赞过该笔记的用户保存到一个集合中,这个集合应该满足什么特点呢?对,就是不能有重复。那么我们是不是就可以联想到redis中的Set或者是SortedSet呢。太好了,这样问题不就解决了吗。
由于接下去我们会实现用户点赞排行榜功能,因此这里选择使用SortedSet来存储用户信息。
@Override
public Result likeBlog(Long id) {
//1. 获取登录用户
UserDTO user = UserHolder.getUser();
if (user == null) {
return Result.fail("用户未登录");
}
Long userId = user.getId();
//2. 判断用户是否点赞
String key = BLOG_LIKED_KEY + id;
Double score = stringRedisTemplate.opsForZSet().score(key, userId.toString());
if (score == null) {
//3. 用户未点赞,可以点赞
//3.1 点赞数+1
boolean flag = update().setSql("liked = liked+1").eq("id", id).update();
if (flag) {
//3.2 将用户加入有序集合中,时间戳代表权重
stringRedisTemplate.opsForZSet().add(key, userId.toString(), System.currentTimeMillis());
}
} else {
//4. 用户已经点赞
boolean flag = update().setSql("liked= liked-1").eq("id", id).update();
if (flag) {
stringRedisTemplate.opsForZSet().remove(key, userId.toString());
}
}
return Result.ok();
}
2.3 代码完善
当我们完成点赞功能时,我们是不是可以添加一个判断用户是否点赞的方法,这样当用户登录后就能看到他是否已经点赞过这条笔记。
//判断博客是否被点赞
private void isBlogLiked(Blog blog) {
UserDTO user = UserHolder.getUser();
if (user == null) {
return;
}
Long userId = user.getId();
String key = BLOG_LIKED_KEY + blog.getId();
Double score = stringRedisTemplate.opsForZSet().score(key, userId.toString());
blog.setIsLike(score != null);
}
接下去,我们是不是也该在查询博客时也判断一下用户是否点赞过
@Override
public Result queryBlogById(Long id) {
Blog blog = this.getById(id);
if (blog == null) {
return Result.fail("笔记不存在");
}
queryBlogUser(blog);
//查询博客是否被点赞
isBlogLiked(blog);
return Result.ok(blog);
}
3. 点赞排行榜
3.1 需求分析
3.2 代码实现
我们先分析一波:首先,我们在用户点赞笔记时是不是已经将用户的id存入了我们的有序集合中呢,有序集合是根据我们传入的时间戳进行排序的。因此,我们调用SortedSet的Range方法是不是就能取出点赞前五名的用户信息,然后传递给前端进行展示呢,对,没错!
//查询博客点赞前五名列表
@Override
public Result queryBlogLikes(Long id) {
String key = BLOG_LIKED_KEY + id;
//1. 查询前top5的用户 ZRange(key,0,4)
Set<String> top5 = stringRedisTemplate.opsForZSet().range(key, 0, 4);
//不加下面可能出现空指针异常
if (top5 == null || top5.isEmpty()) {
return Result.ok(Collections.emptyList());
}
//2. 解析用户的id
List<Long> ids = top5.stream().map(Long::valueOf).collect(Collectors.toList());
String idStr = StrUtil.join(",", ids);
//3. 根据用户id查询用户 where id in(2,1) order by field(id,2,1)即按时间顺序排列
List<UserDTO> userDTOS = userService.query()
.in("id", ids).last("ORDER BY FIELD(id," + idStr + ")").list()
.stream()
.map(user -> BeanUtil.copyProperties(user, UserDTO.class))
.collect(Collectors.toList());
//4. 返回用户对象
return Result.ok(userDTOS);
}
OK,到这里我们就完成了本篇blog所期望完成功能。这些功能也都是基于redis相关的数据集合下完成的,加油,好好学习!!!
这些功能其实很早之前就写完了,应该早就该发了,但是由于其他一些原因,一拖再拖,害…都是借口,呜呜呜…