自己写一个类,继承CountDowmTimer
public class VerificationCountDownTimer extends CountDownTimer
复制代码
01 构造函数
/**
* 类中的构造函数
* @param millisInFuture
* @param countDownInterval
*/
public VerificationCountDownTimer(long millisInFuture,long countDownInterval){
super(millisInFuture, countDownInterval);
}
复制代码
- 两个参数(均已毫秒为单位)
- millisInFuture 倒计的总时间
- countDownInterval 倒计的时间间隔
- 以 60s 的倒计时为例?,如下使用
verificationCountDownTimer = new VerificationCountDownTimer( 60*1000 , 1000)
复制代码
02 方法
public final void cancel ()
public abstract void onFinish ()
public abstract void onTick (long millisUntilFinished)
public final CountDownTimer start ()
复制代码
-
cancel() 取消当前任务
-
onFinish() 结束的时候调用
-
onTick(long millisUntilFinished) 当前任务每完成一次倒计时间间隔时调用,直接对UI进行更新
- 这个方法中的参数是在倒计时过程中传入进来的毫秒数,比如倒计时总时间为60秒,时间间隔为1秒,那么这个参数的传进来的值依次为:59、58、57、56.....1
-
start() 开始当前任务
03 开发过程中的注意点
手机发送验证码需要花费1条短信的费用0.10-0.15元才能获取验证码,平台当然就希望尽可能多的省钱,所以开发端就需要设置验证码倒计时,防止用户出于有趣或者恶意一直在点击获取新验证码,这样就大大加重了平台的运营负担。所以加上验证码倒计时,不仅能防止用户心急戳戳戳,或因网络延迟而造成请求堆积,也能帮公司省点小钱。知道了使用原因,那就需要考虑一些特殊情形下的处理,比如:
- 当验证码进入倒计时,点击后退,再重新进入有验证码的界面,验证码是重新start呢还是继续处于前一个验证码倒计时状态呢?
- 解决方法:设置一个检测量,利用 true 和 false 来进行判断(在自己写的类里面,先定义一个检测量,然后在timerStart 内部进行设置)
VerificationCountDownTimer.java
public class VerificationCountDownTimer extends CountDownTimer {
public static long curMillis =0;
public static boolean FLAG_FIRST_IN =true;
/**
* 类中的构造函数
* @param millisInFuture
* @param countDownInterval
*/
public VerificationCountDownTimer(long millisInFuture,long countDownInterval){
super(millisInFuture, countDownInterval);
}
/**
* 当前任务每完成一次倒计时间隔时间时回调
* @param millisUntilFinished
*/
@Override
public void onTick(long millisUntilFinished) {
}
/**
* 当前任务完成的时候回调
*/
@Override
public void onFinish() {
}
public void timerStart(boolean onClick){
if(onClick) {
VerificationCountDownTimer.curMillis= System.currentTimeMillis();
}
VerificationCountDownTimer.FLAG_FIRST_IN=false;
this.start();
}
}
复制代码
一旦使用 timeeStart的时候,就会将 检测量设置为 false ,然然后在可以在 另一个Activity中 作为条件 进行判断使用
RegisterActivity.java 关键部分(绑定按钮的点击事件)
- initCountDownTimer
- 如果满足“不是第一次进入” 且 “倒计时时间不为0”的条件的话, 启动 setCountDownTimer 方法,并将 检测器 设置为false
衡量“倒计时时间是否为0”的方法
- 在点击Button的时候,启动的timerStart中,就将 curMillis 获取为当前时间,所以判断依据为:curMillis + 60000 ms 是否大于 当前时间
- 大于: 启动 setCountDownTimer,setCountDownTimer内置onTick方法,对获取得到的剩余时间不断的进行UI的展示
补充:system.currentTimeMillis() 获得当前的时间
该方法的作用是返回当前的计算机时间,单位:毫秒。之前使用new Date()来获取当前时间,new Date()所做的事情其实就是调用了System.currentTimeMillis()。
最终实现的效果图
1、 初始界面
2、 点击获取验证码后
3、 后退其他界面,再次进入,若60s未结束,则还是处于倒计时中
4、 60s 结束,可以点击 重新获取 验证码
5、 点击后,再次进入倒计时
最清晰的解释,先来看郑哥Activity的关键代码部分
public class RegisterActivity extends BaseNoBarPresenterActivity {
int tag =1;
VerificationCountDownTimer verificationCountDownTimer;
@BindView(R.id.img_agreement_selector)
ImageView imgAgreementSelector;
@Override
protected BaseContract.Presenter initPresenter() {
return new BasePresenter<>(this);
}
public static void startActivity(Context context) {
context.startActivity(new Intent(context, RegisterActivity.class));
}
@Override
protected void initVariable() {
initCountDownTimer();
}
@Override
protected void initView() {
}
@Override
protected void loadData() {
}
@OnClick(R.id.btn_register)
public void onBtnRegisterClicked() {
ToastUtil.showToast("还没有网络请求,敬请期待");
LoginActivity.startActivity(RegisterActivity.this);
verificationCountDownTimer.cancel();
}
@OnClick(R.id.img_agreement_selector)
public void onViewClicked() {
tag ++;
if(tag%2==0)//说明遇到点击为已阅读
{
imgAgreementSelector.setSelected(true);
}
if (tag %2!=0)
{
imgAgreementSelector.setSelected(false);
}
}
/**
* 忘记密码的返回按钮,从忘记密码界面,回到登录界面
*/
@OnClick(R.id.ly_header_register)
public void onLyHeaderRegisterClicked() {
// LoginActivity.startActivity(RegisterActivity.this);
this.finish();
}
/**
* 绑定 获得验证码 按钮,启动timerStart
*/
@OnClick(R.id.btn_verification_register)
public void onBtnVerificationRegisterClicked() {
ToastUtil.showToast("你即将获得验证码,请注意查收");
verificationCountDownTimer.timerStart(true);
}
/**
* 倒计时具体方法
*/
public void initCountDownTimer() {
if(!VerificationCountDownTimer.FLAG_FIRST_IN&&
VerificationCountDownTimer.curMillis+60000>System.currentTimeMillis()) {
setCountDownTimer(VerificationCountDownTimer.curMillis+60000-System.currentTimeMillis());
verificationCountDownTimer.timerStart(false);
} else {
setCountDownTimer(60000);
}
}
public void setCountDownTimer(final long countDownTime) {
verificationCountDownTimer = new VerificationCountDownTimer( countDownTime , 1000) {
@Override
public void onTick(long millisUntilFinished) {
btnVerificationRegister.setEnabled(false);
btnVerificationRegister.setText((millisUntilFinished / 1000) + " s");
}
@Override
public void onFinish() {
btnVerificationRegister.setEnabled(true);
btnVerificationRegister.setText(getString(R.string.btn_verification_gain));
if(countDownTime!=60000) {
setCountDownTimer(60000);
}
}
};
}
}
复制代码
- initCountDownTimer();方法是在一开始就进行了初始化调用
- 所有遇到的困扰都在这里进行了解释