Appearance
Set结构是一个无序不重复集合,用来存储黑名单信息再合适不过,还能通过原生的API查询某数据是否在集合中。
TIP
场景分析:淘宝的商品评价功能,不是任何人就能评价的,有一种职业就是差评师。差评师就是勒索敲诈商家,这种差评师在淘宝里面就被设置了黑名单,即使购买了商品,也评价不了。
TIP
redis技术解决方案:黑名单过滤器除了针对上文说的淘宝评价,针对用户黑名单外,其实还有ip黑名单、设备黑名单等。在高并发的情况下,通过数据库过滤明显不符合要求,一般的做法都是通过Redis来实现的。
步骤1:先把数据库的数据同步到redis的set集合中。
java
/**
* 向Redis模拟存储黑名单数据
* @author cv大魔王
* @version 1.0
* @date 2021/5/23 14:16
*/
@Component
public class TaskService {
@Autowired
private RedisTemplate redisTemplate;
/**
* -@PostConstruct该注解被用来修饰一个非静态的void()方法。
* 被@PostConstruct修饰的方法会在服务器加载Servlet的时候运行,并且只会被服务器执行一次。
* PostConstruct在构造函数之后执行,init()方法之前执行。
*/
@PostConstruct
public void init() {
List<Integer> blankList = this.blackList();
blankList.forEach(t-> redisTemplate.opsForSet().add("blank:user",t));
}
/**
* 模拟100个黑名单
* 实际场景应防止到数据库中
* @return
*/
public List<Integer> blackList() {
ArrayList<Integer> list = new ArrayList<>();
for (int i = 0; i < 100; i++) {
list.add(i);
}
return list;
}
}
步骤2:评价的时候验证是否为黑名单,通过sismember命令来实现。
java
@RestController
@RequestMapping("/api/blank")
public class BlankController {
@Autowired
private RedisTemplate redisTemplate;
/**
* 黑名单校验接口
* @param userId 用户id 实际生产场景可使用token判断
* @return true 黑名单
*/
@GetMapping("/isBlacklist")
public boolean isBlankList(Integer userId) {
boolean flag = false;
try {
flag = this.redisTemplate.opsForSet().isMember("blank:user", userId);
} catch (Exception e) {
//这里的异常,一般是redis瘫痪,或redis网络timeoutlog.error( "exception: " , ex) ;
// TODO 走DB查询
}
return flag;
}
}