iOS 计时器 单例 GCD 切换控制器倒计时继续读秒

       

       写项目的时候需要一个计时器来做短信验证码的倒计时,场景需求是按钮点击后开始读秒,按钮不可用,当读秒结束按钮可用,在读秒期间切换控制器,或者将应用切换到后台在切换回读秒页面依旧在继续读秒。

代码如下:
 一个继承NSObject的单例类

.h文件

#import <Foundation/Foundation.h>

@interface CaptchaTimerManager : NSObject

@property (nonatomic, assign)__block int timeout;

+ (id)sharedTimerManager;

- (void)countDown;

@end


.m文件

#import "CaptchaTimerManager.h"

@implementation CaptchaTimerManager


+ (id)sharedTimerManager{

    static CaptchaTimerManager *manager = nil;

    static dispatch_once_t onceToken;

    dispatch_once(&onceToken, ^{

        if (manager == nil) {

            manager = [[self alloc]init];

        }

    });

    return manager;

}


- (void)countDown{

    if (_timeout > 0) {

        dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

        dispatch_source_t _timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0,queue);

        dispatch_source_set_timer(_timer,dispatch_walltime(NULL, 0),1.0*NSEC_PER_SEC, 0); //每秒执行

        dispatch_source_set_event_handler(_timer, ^{

            if(_timeout<=0){ //倒计时结束,关闭

                dispatch_source_cancel(_timer);

            }else{

                _timeout--;

            }

        });

        dispatch_resume(_timer);

    }

}

@end


在实际需要计时器的控制器中具体实现细节如下

#import "ViewController.h"

#import "CaptchaTimerManager.h"


@interface ViewController ()

@property (strong, nonatomic)UIButton *button;

@property (nonatomic, assign) __block int timeout;

@end


@implementation ViewController

//页面出现前取出计时器单例的时间进行判断是否还在倒计时

- (void)viewWillAppear:(BOOL)animated{

    [super viewWillAppear:animated];

    CaptchaTimerManager *manager = [CaptchaTimerManager sharedTimerManager];

    int temp = manager.timeout;

        if (temp > 0) {

            _timeout= temp; //倒计时时间

            [self timerCountDown];

        }

}

//页面消失前记录倒计时时间到单例里

- (void)viewWillDisappear:(BOOL)animated{

    [super viewWillDisappear:animated];

    if (self.timeout > 0) {

        CaptchaTimerManager *manager = [CaptchaTimerManager sharedTimerManager];

        if (manager.timeout == 0) {

            manager.timeout = _timeout;

            [manager countDown];

        }

        _timeout = 0;//置为0,释放controller

    }

}


//按钮点击事件

- (IBAction)touch1:(id)sender {

    _timeout = 60; //倒计时时间

    [self timerCountDown];

   

}


//控制器里的计时器

- (void)timerCountDown{

    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

    dispatch_source_t _timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0,queue);

    dispatch_source_set_timer(_timer,dispatch_walltime(NULL, 0),1.0*NSEC_PER_SEC, 0); //每秒执行

    dispatch_source_set_event_handler(_timer, ^{

        if(_timeout<=0){ //倒计时结束,关闭

            dispatch_source_cancel(_timer);

            dispatch_async(dispatch_get_main_queue(), ^{

                //这里写倒计时结束button的处理

            });

        }else{

            dispatch_async(dispatch_get_main_queue(), ^{

                //这里写倒计时期间button的处理(重设button的tiitle、用户交互等)

            });

            _timeout--;

        }

    });

    dispatch_resume(_timer);

}

- (void)viewDidLoad {

    [super viewDidLoad];

    // button初始化处理.

}

@end


原理大概就是这样子,按钮细节自行调整就好


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
由于苹果官方对iOS系统后台任务管控非常严格,所以在iOS系统要实现倒计时进入后台或者切换页面之后依然能保持倒计时状态会比较繁琐或困难,但倒计时按钮是经常用到、司空见惯!经常看到不太友好的倒计时按钮表现为,倒计时并未完成但当你切换页面或者进入后台后按钮倒计时失效并又可以点击了,实际上没有达到限制1~2分钟后才可点击的效果。于是我琢磨下封装了这个倒计时按钮的SDK它能解决上述问题并完美达到预期效果,而且SDK支持代码创建、XIB创建,使用也非常简单。注意这个SDK只能在真机上面跑,不能在xcode模拟器上面跑,因为时间匆忙没有做模拟器兼容SDK,请在真机上面使用SDK,下次有时间再更新。 SDK提供有比较灵活的属性控制,可修改倒计时按钮背景颜色、文字颜色、按钮圆角大小、点击时背景颜色,基本能满足全部UI设计需求。请下载资源包里面提供了SDK的用法截图、Demo、SDK(务必注意看看 5张SDK使用说明图片) 用法: 请直接把“SDK”文件夹拖到您的项目中,然后安照demo代码应用SDK即可,不明白请联系我们 注意1: 项目中有多个倒计时按钮,那么必须给按钮“indexsButtonLDS”设置不同的标识符(任意字符串),否则会出现异常。代码或者XIB方式设置indexsButtonLDS请参考代码,比较简单的(如果不明白仔细看下Demo) 注意2: 倒计时最长为180秒(因受iOS系统限制),超过180秒会无效。通常倒计时按钮都是限制2分钟吧。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值