BBS论坛项目相关-9:Redis点赞模块
点赞
对于某个实体的赞,key对应用entityType和entityId拼接PREFIX_ENTITY_LIKE + SPLIT + entityType + SPLIT + entityId
对于实体的赞采用set数据结构,field为用户id,可用于记录对该实体点赞的用户,此外还需要记录该用户是否已经对该实体点过赞,点过了再点就会取消点赞。
对于某个用户的点赞:key为用户id进行拼接PREFIX_USER_LIKE + SPLIT + userId
,
对于用户的点赞只需要用String结构存储,多用于统计用户收到的点赞数量
likeService层:
用户点赞
key为entityType和entityId拼接,field为点赞userid
用set数据类型存储,先判断以该key值和该用户id的field的对象是否存在,存在就已经点过赞了,点过了就移除这个member,取消点赞。
查看用户收到的赞,如果当前用户对该实体点赞,该实体的发布者即entityUserId收到的赞加一,取消赞则减一,所以传递参数时需要传递两个userID,一个是当前持有的用户id,即自己本身,另一个是发布该实体的作者用户id。
对点赞功能进行重构:
这两件事为一个事务,需要同时进行。由于事务中查询不起作用要注意将查询语句放在开启事务之前。
public void like(int userId, int entityType, int entityId, int entityUserId) {
redisTemplate.execute(new SessionCallback() {
@Override
public Object execute(RedisOperations operations) throws DataAccessException {
String entityLikeKey = RedisKeyUtil.getEntityLikeKey(entityType, entityId);
String userLikeKey = RedisKeyUtil.getUserLikeKey(entityUserId);
boolean isMember = operations.opsForSet().isMember(entityLikeKey, userId);
operations.multi();
if (isMember) {
operations.opsForSet().remove(entityLikeKey, userId);
operations.opsForValue().decrement(userLikeKey);
} else {
operations.opsForSet().add(entityLikeKey, userId);
operations.opsForValue().increment(userLikeKey);
}
return operations.exec();
}
});
}
查询某实体点赞的数量
public long findEntityLikeCount(int entityType, int entityId) {
String entityLikeKey = RedisKeyUtil.getEntityLikeKey(entityType, entityId);
return redisTemplate.opsForSet().size(entityLikeKey);
}
查询某人对某实体是否点过赞
public int findEntityLikeStatus(int userId, int entityType, int entityId) {
String entityLikeKey = RedisKeyUtil.getEntityLikeKey(entityType, entityId);
return redisTemplate.opsForSet().isMember(entityLikeKey, userId) ? 1 : 0;
}
查询某个用户获得的赞
public int findUserLikeCount(int userId) {
String userLikeKey = RedisKeyUtil.getUserLikeKey(userId);
Integer count = (Integer) redisTemplate.opsForValue(