登录信息设计
isfail方法逻辑:
class Person{
//重置时间--16s
private static final int RESET_TIME=16;
//密码连续输入3次失败的持续时间--8
private static final int DURATION=8;
//最大输入失败次数3
private static final int MAX_TIMES=3;
//用户id
private String id;
//登录失败次数
private int fail;
//第一次失败的时间
private long firstFail;
//时间链表
private LinkedList<Long> times;
//当前用户是否锁定
private boolean lock;
public Person() {
}
public Person(String id, int fail, long firstFail, LinkedList<Long> times, boolean lock) {
this.id = id;
this.fail = fail;
this.firstFail = firstFail;
this.times = times;
this.lock = lock;
}
public static int getResetTime() {
return RESET_TIME;
}
public static int getDURATION() {
return DURATION;
}
public static int getMaxTimes() {
return MAX_TIMES;
}
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public int getFail() {
return fail;
}
public void setFail(int fail) {
this.fail = fail;
}
public long getFirstFail() {
return firstFail;
}
public void setFirstFail(long firstFail) {
this.firstFail = firstFail;
}
public LinkedList<Long> getTimes() {
return times;
}
public void setTimes(LinkedList<Long> times) {
this.times = times;
}
public boolean isLock() {
return lock;
}
public void setLock(boolean lock) {
this.lock = lock;
}
//密码输入错误进入此方法
public void isFailed(){
long now=System.currentTimeMillis()/1000;
System.out.println("第一次登录失败的时间是"+(now-firstFail)+"秒前");
//如果超过了三十分钟,需要进行重置
if(now>firstFail+RESET_TIME){
System.out.println("你重新获得了"+MAX_TIMES+"次机会");
fail=1;//失败次数修改为1,因为这一次已经失败
firstFail=now;
times=new LinkedList<>();
this.lock=false;
return;
}else{
//如果已经是锁定状态了
if(lock){
System.out.println("账户已经锁定了,请等待"+(firstFail+RESET_TIME-now)+"秒之后再来");
return;
}
//之前记录的第一次登录失败时间在10分钟之前了,需要更换
while (!times.isEmpty()&&now>times.getFirst()+DURATION){
times.removeFirst();
fail--;
firstFail=times.isEmpty()?now:times.getFirst();//如果是空的说明没有记录那就用当前时间
}
if(fail>=MAX_TIMES&&now<firstFail+DURATION){
System.out.println("连续3次密码错误,登录失败");
lock=true;
}else if(fail<MAX_TIMES){
fail++;
System.out.println("密码错误"+fail+"次");
}
}
//不管怎样当前都是一个错误的登入次数,需要加入链表
times.addLast(now);
}
}
接口设计:
用一个map存储所有登录的用户模拟数据库
private static Map<String, Person> map = new HashMap<>();//记录登录错误的人的信息
public static void login(String id, boolean flag){
if (flag){
// 登陆成功
return;
}else{
Person p = null;
// 登录失败
synchronized (map) {
p = map.get(id);
if (p == null){
p = new Person(id, 0, System.currentTimeMillis() / 1000,
new LinkedList<>(), false);
map.put(id, p);
return;
}
p.isFailed();
}
}
}
接口测试
public static void main(String[] args) {
for(int i = 0; i < 20; i ++){
try{ TimeUnit.SECONDS.sleep(1); }catch (InterruptedException e){ e.printStackTrace(); }
login("aaa", false);
}
}
结果:
第一次登录失败的时间是1秒前 密码错误1次 第一次登录失败的时间是2秒前 密码错误2次 第一次登录失败的时间是3秒前 密码错误3次 第一次登录失败的时间是4秒前 连续3次密码错误,登录失败 第一次登录失败的时间是5秒前 账户已经锁定了,请等待11秒之后再来 第一次登录失败的时间是6秒前 账户已经锁定了,请等待10秒之后再来 第一次登录失败的时间是7秒前 账户已经锁定了,请等待9秒之后再来 第一次登录失败的时间是8秒前 账户已经锁定了,请等待8秒之后再来 第一次登录失败的时间是9秒前 账户已经锁定了,请等待7秒之后再来 第一次登录失败的时间是10秒前 账户已经锁定了,请等待6秒之后再来 第一次登录失败的时间是11秒前 账户已经锁定了,请等待5秒之后再来 第一次登录失败的时间是12秒前 账户已经锁定了,请等待4秒之后再来 第一次登录失败的时间是13秒前 账户已经锁定了,请等待3秒之后再来 第一次登录失败的时间是14秒前 账户已经锁定了,请等待2秒之后再来 第一次登录失败的时间是15秒前 账户已经锁定了,请等待1秒之后再来 第一次登录失败的时间是16秒前 账户已经锁定了,请等待0秒之后再来 第一次登录失败的时间是17秒前 你重新获得了5此机会 第一次登录失败的时间是1秒前 密码错误2次 第一次登录失败的时间是2秒前 密码错误3次
|