我正在参与「启航计划」
背景
最近分配到一个需求,要生成6位数验证码,运用验证码做一系列操作。由于需求运用验证码去确定用户信息,所以有唯一性要求。
计划一:随机函数Random()
运用随机函数生成1000000以内的数,然后再补齐位数,然后校验唯一性,假如已存在就在生成一个随机数。
/**
* 计划一:随机函数Random()
*/
@Test
public void testRandom() {
Random rd = new Random();
for (int i = 0; i < 10; i++) {
val numberCode = rd.nextInt(1000000);
System.out.println(String.format("%06d", numberCode));
}
}
输出结果:
731311
508327
383928
551535
623440
758321
284932
039829
365014
500294
Process finished with exit code 0
优点:
- 完成简单快捷
缺点:
- 重复概率相对较大
计划二:Redis令牌桶方式
数字码的生成首要考虑效率问题,假如运用随机函数random()随机生成1000000以内的随机数,可能比较简单呈现数据重复问题。因而我们先初始化将000000-999999总的100万个数据先放到redis中,然后随机获取一个,等到redis中没有数据再从头初始化100万个数据,这样理论上短时间内不会呈现已存在的数据。
private final String KEY_NUMBER_CODE = "visitor-number-code";
@Autowired
private RedisTemplate<String, String> redisTemplate;
public String getNumberCode(String key) {
String numberCode = redisTemplate.opsForSet().pop(key);
if (StringUtils.isBlank(numberCode)) {
// 运用Redis分布式锁操控并发
numberCode = initNumberCode(key);
}
return numberCode;
}
/**
* Redis初始化1000000个数据
*/
public String initNumberCode(String key) {
// 双重查看,防止多个进行一起在等候分布式锁初始化
String numberCode = redisTemplate.opsForSet().pop(key);
if (StringUtils.isNotBlank(numberCode)) {
return numberCode;
}
// 将数据初始化到数组再一次性放到Redis中,提高功能
List<String> list = new ArrayList<>();
for (int i = 1; i < 1000000; i++) {
list.add(String.format("%06d", i));
}
String[] arr = list.toArray(new String[0]);
redisTemplate.opsForSet().add(key , arr);
return redisTemplate.opsForSet().pop(key);
}
代码完成关键:
- 运用Redis分布式锁操控并发
- 双重查看,防止多个进行一起在等候分布式锁初始化
- 将数据初始化到数组再一次性放到Redis中,提高功能
补偿措施
我们要赶快减少数字码呈现重复的概率,假如呈现重复数据要有重试机制,从头获取数字码。
由于6位数字码是从000000-999999只有100万个数据,假如不及时铲除数字码的话,日积月累总会用完的,因而要拟定失效铲除战略。或者运用字符组合,构成更加丰富的验证码。