Android 验证码倒计时实现方式总结

   2016-11-03 0
核心提示:Android验证码倒计时实现方式总结几乎所有的APP里面都有验证码倒计时按钮,自己在项目中也尝试,简单总结为以下三种:1、使用线程和Handler的方式,定时刷新倒计时数字,这种方式容易导致内存泄露,所以一般都使用弱引用,控制数字的刷新。2、自定义倒计时按

Android验证码倒计时实现方式总结

几乎所有的APP里面都有验证码倒计时按钮,自己在项目中也尝试,简单总结为以下三种:

1、使用线程和Handler的方式,定时刷新倒计时数字,这种方式容易导致内存泄露,所以一般都使用弱引用,控制数字的刷新。

2、自定义倒计时按钮

3、使用Android提供的CountDownTimer结合TextView实现倒计时功能

方式一是我刚开始做android的时候,因为那个时候android只学了个基础,学会了使用线程结合Handler的方式更新视图,具体实现如下:

private static class MyHandler extends android.os.Handler{

WeakReference<MainActivity> mActivity ;

public MyHandler(MainActivity activity) {

mActivity = new WeakReference<>(activity);

}

@Override

publicvoid handleMessage(Message msg) {
//

TODO Auto-generated method stub

final
MainActivity mf = mActivity .get();

super .handleMessage(msg);

switch (msg. what ) {

case 0 :

mf. sendOne .setText( " 重新发送 " );

mf. count = 60 ;

mf. sendOne .setEnabled( true );

break ;

case 1 :

mf. sendOne .setEnabled( false );

mf. sendOne .setText(mf. count + " " );

break ;

}

}

}

private void sendCode(){

final Runnable r = new Runnable() {

publicvoid run() {

runningOne = true ;

while ( count > 0 ) {

count = count - 1 ;

handler .sendEmptyMessage( 1 );

try {

Thread. sleep ( 1000 );

} catch (InterruptedException e) {

//

TODO Auto-generated catch block

e.printStackTrace();

}

}

runningOne = false ;

handler .sendEmptyMessage( 0 );

}

};

Thread s = new Thread(r);

s.start();

}

当然在使用这种方式的时候,我们还必须保证在倒计时期间,按钮是不能点击的,所以需要定义一个标志变量

if (runningOne) {
} else {
    sendCode();
}

方式二:使用自定义按钮实现倒计时功能

public class CountButton extends Button implements View.OnClickListener {


    /*
     倒计时时长,默认计时时间
     */
    private long defaultTime = 60*1000;
    private long time = defaultTime;

    /*
    开始执行计时的类,可以每间隔一秒执行任务
     */
    private Timer timer;

    /*
    执行的任务
     */
    private TimerTask task;

    /*
    默认文案
     */
    private String defaultText = "获取验证码";
    /*
    计时完成之后显示的文案
     */
    private String finishText = "重新发送";


    /*
    点击事件
     */
    private OnClickListener onClickListener;

    public CountButton(Context context) {
        super(context);
        initView();
    }

    public CountButton(Context context, AttributeSet attrs) {
        super(context, attrs);
        initView();
    }

    public CountButton(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        initView();
    }


    private void initView() {
        if (!TextUtils.isEmpty(getText())) {
            defaultText = getText().toString().trim();
        }
        this.setText(defaultText);
        setOnClickListener(this);
    }

    /*
    更新显示文案
     */
    private Handler handler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            CountButton.this.setText(time / 1000 + " ");
            time -= 1000;
            if (time < 0) {
                CountButton.this.setEnabled(true);
                CountButton.this.setText(finishText);
                clearTimer();
                time = defaultTime;
            }
        }
    };

    /*
    清除倒计时
     */
    private void clearTimer() {
        if (task != null) {
            task.cancel();
            task = null;
        }
        if (timer != null) {
            timer.cancel();
            timer = null;
        }
    }

    /*
    初始化时间
     */
    private void initTimer() {
        timer = new Timer();
        task = new TimerTask() {
            @Override
            public void run() {
                handler.sendEmptyMessage(1);
            }
        };
    }


/*
activity或者fragment消亡的时候清除倒计时
 */

    @Override
    protected void onDetachedFromWindow() {
        clearTimer();
        super.onDetachedFromWindow();
    }


    public void setDefaultText(String defaultText) {
        this.defaultText = defaultText;
    }


    public void setFinishText(String finishText) {
        this.finishText = finishText;
    }


    public void setDefaultTime(long defaultTime) {
        this.defaultTime = defaultTime;
    }

    @Override
    public void onClick(View view) {
        start();
        if (onClickListener != null) {
            onClickListener.onClick(view);
        }
    }

    public void start() {
        initTimer();
        this.setText(time / 1000 + " ");
        this.setEnabled(false);
        timer.schedule(task, 0, 1000);
    }

    @Override
    public void setOnClickListener(OnClickListener l) {
        if (l instanceof CountButton) {
            super.setOnClickListener(l);
        } else {
            this.onClickListener = l;
        }
    }
}

为了实现方便,我没有使用attrs声明相关的属性,源码不多,需要使用的小伙伴可以直接在源码中修改

setDefaultTime( long defaultTime)方法设置倒计时总的计时时间,例如:sendTwo.setDefaultTime( 60 * 1000 );

setDefaultText(String defaultText)方法设置点击按钮之前显示的文案,例如:sendTwo.setDefaultText("发送验证码");
setFinishText(String finishText)方法设置倒计时完成之后显示的文案,例如:sendTwo.setFinishText("重新发送");
方法三:使用Android提供的CountDownTimer的onTick(long l) 和onFinish()方法控制计时中和结束后的相关操作,这种方式可以适用于很多倒计时的场景,个人觉得比较方便、适用。
private CountDownTimer downTimer = new CountDownTimer(60 * 1000, 1000) {
    @Override
    public void onTick(long l) {
        runningThree = true;
        sendThree.setText((l / 1000) + "");
    }

    @Override
    public void onFinish() {
        runningThree = false;
        sendThree.setText("重新发送");
    }
};
然后使用downTimer.start()开始倒计时。
使用这种方式和第一种方式一样,我们也需要定义标志位来控制按钮的点击事件
if (runningThree) {
} else {
    downTimer.start();
}
上面就是我在开发的过程中所接触过的倒计时方式,简单总结一下:
方式一,可以说是最复杂,也是最笨重的方式,这种方式容易引起内存泄露,及时使用了弱引用的方式,也避免不了内存泄露的可能。
方式二,比较简单,只需要设置好一些需要用到的字段和数值,就能够完成倒计时功能,方便、快捷。
方式三,简单,使用场合比较广泛,可以替代TimerTask,完成定时执行任务的操作,比较方便、适用。
源码我已经上传到CSDN上面,感兴趣的小伙伴可以下载看看,传送门开启:http://download.csdn.net/detail/zhimingshangyan/9670286
 
标签: 安卓开发
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • 安卓中通知功能的具体实现
    安卓中通知功能的具体实现
    通知[Notification]是Android中比较有特色的功能,当某个应用程序希望给用户发出一些提示信息,而该应用程序又不在前台运行时,就可以借助通知实现。使用通知的步骤1、需要一个NotificationManager来获得NotificationManager manager = (NotificationManager
    02-05 安卓开发
  • Android view系统分析-setContentView
    Android view系统分析-setContentView
    第一天上班,列了一下今年要学习的东西。主要就是深入学习Android相关的系统源代码,夯实基础。对于学习Android系统源代码,也没什么大概,就从我们平常使用最基础的东西学起,也就是从view这个切入点开始学习Android的源码,在没分析源码之前,我们有的时候
    02-05 安卓开发
  • 如何进行网络视频截图/获取视频的缩略图
    如何进行网络视频截图/获取视频的缩略图
    小编导读:获取视频的缩略图,截图正在播放的视频某一帧,是在音视频开发中,常遇到的问题。本文是主要用于点播中截图视频,同时还可以获取点播视频的缩略图进行显示,留下一个问题,如下图所示, 如果要获取直播中节目视频缩略图,该怎么做呢?(ps:直播是直
  • Android NDK 层发起 HTTP 请求的问题及解决
    Android NDK 层发起 HTTP 请求的问题及解决
    前言新的一年,大家新年快乐~~鸡年大吉!本次给大家带来何老师的最新文章~虽然何老师还在过节,但依然放心不下广大开发者,在此佳节还未结束之际,给大家带来最新的技术分享~ 事件的起因不说了,总之是需要实现一个 NDK 层的网络请求。为了多端适用,还是选择
  • Android插件化(六): OpenAtlasの改写aapt以防止资源ID冲突
    Android插件化(六): OpenAtlasの改写aapt以防
    引言Android应用程序的编译中,负责资源打包的是aapt,如果不对打包后的资源ID进行控制,就会导致插件中的资源ID冲突。所以,我们需要改写aapt的源码,以达到通过某种方式传递资源ID的Package ID,通过aapt打包时获取到这个Package ID并且应用才插件资源的命名
    02-05 安卓开发
  • Android架构(一)MVP架构在Android中的实践
    Android架构(一)MVP架构在Android中的实践
    为什么要重视程序的架构设计 对程序进行架构设计的原因,归根结底是为了 提高生产力 。通过设计是程序模块化,做到模块内部的 高聚合 和模块之间的 低耦合 (如依赖注入就是低耦合的集中体现)。 这样做的好处是使得程序开发过程中,开发人员主需要专注于一点,
    02-05 安卓开发
  • 安卓逆向系列教程 4.2 分析锁机软件
    安卓逆向系列教程 4.2 分析锁机软件
    安卓逆向系列教程 4.2 分析锁机软件 作者: 飞龙 这个教程中我们要分析一个锁机软件。像这种软件都比较简单,完全可以顺着入口看下去,但我这里还是用关键点来定位。首先这个软件的截图是这样,进入这个界面之后,除非退出模拟器,否则没办法回到桌面。上面那
    02-05 安卓开发
  • Android插件化(二):OpenAtlas插件安装过程分析
    Android插件化(二):OpenAtlas插件安装过程分析
    在前一篇博客 Android插件化(一):OpenAtlas架构以及实现原理概要 中,我们对应Android插件化存在的问题,实现原理,以及目前的实现方案进行了简单的叙述。从这篇开始,我们要深入到OpenAtlas的源码中进行插件安装过程的分析。 插件的安装分为3种:宿主启动时立
    02-05 安卓开发
  • [译] Android API 指南
    [译] Android API 指南
    众所周知,Android开发者有中文网站了,API 指南一眼看去最左侧的菜单都是中文,然而点进去内容还是很多是英文,并没有全部翻译,我这里整理了API 指南的目录,便于查看,如果之前还没有通读,现在可以好好看一遍。注意,如果标题带有英文,说明官方还没有翻
  • 使用FileProvider解决file:// URI引起的FileUriExposedException
    使用FileProvider解决file:// URI引起的FileUri
    问题以下是一段简单的代码,它调用系统的相机app来拍摄照片:void takePhoto(String cameraPhotoPath) {File cameraPhoto = new File(cameraPhotoPath);Intent takePhotoIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);takePhotoIntent.putExtra(Medi
    02-05 安卓开发
点击排行