zoukankan      html  css  js  c++  java
  • 使用 GCD 实现倒计时效果

    效果如下:

    ViewController.h

    1 #import <UIKit/UIKit.h>
    2 
    3 @interface ViewController : UIViewController
    4 @property (assign, nonatomic) NSInteger surplusSecond;
    5 
    6 @property (strong, nonatomic) IBOutlet UILabel *lblMessage;
    7 @property (strong, nonatomic) IBOutlet UIButton *btnSendCAPTCHA;
    8 
    9 @end

    ViewController.m

     1 #import "ViewController.h"
     2 
     3 @interface ViewController ()
     4 - (void)layoutUI;
     5 - (void)countDown;
     6 @end
     7 
     8 @implementation ViewController
     9 #define kSurplusSecond 5
    10 
    11 - (void)viewDidLoad {
    12     [super viewDidLoad];
    13     
    14     [self layoutUI];
    15 }
    16 
    17 - (void)didReceiveMemoryWarning {
    18     [super didReceiveMemoryWarning];
    19     // Dispose of any resources that can be recreated.
    20 }
    21 
    22 - (void)layoutUI {
    23     _surplusSecond = kSurplusSecond; //剩余秒数;这里指验证码发送完,间隔多少秒才能再次点击「验证」按钮进行发送验证码
    24     
    25     _btnSendCAPTCHA.tintColor = [UIColor darkGrayColor];
    26     _btnSendCAPTCHA.layer.masksToBounds = YES;
    27     _btnSendCAPTCHA.layer.cornerRadius = 10.0;
    28     _btnSendCAPTCHA.layer.borderColor = [UIColor grayColor].CGColor;
    29     _btnSendCAPTCHA.layer.borderWidth = 1.0;
    30 }
    31 
    32 /**
    33  *  倒计时
    34  */
    35 - (void)countDown {
    36     //全局并发队列
    37     dispatch_queue_t globalQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    38     //主队列;属于串行队列
    39     dispatch_queue_t mainQueue = dispatch_get_main_queue();
    40     
    41     //定时循环执行事件
    42     //dispatch_source_set_timer 方法值得一提的是最后一个参数(leeway),他告诉系统我们需要计时器触发的精准程度。所有的计时器都不会保证100%精准,这个参数用来告诉系统你希望系统保证精准的努力程度。如果你希望一个计时器每5秒触发一次,并且越准越好,那么你传递0为参数。另外,如果是一个周期性任务,比如检查email,那么你会希望每10分钟检查一次,但是不用那么精准。所以你可以传入60,告诉系统60秒的误差是可接受的。他的意义在于降低资源消耗。
    43     dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, globalQueue);
    44     dispatch_source_set_timer(timer, DISPATCH_TIME_NOW, 1.0 * NSEC_PER_SEC, 0.0 * NSEC_PER_SEC);
    45     dispatch_source_set_event_handler(timer, ^{ //计时器事件处理器
    46         NSLog(@"Event Handler");
    47         if (_surplusSecond <= 0) {
    48             dispatch_source_cancel(timer); //取消定时循环计时器;使得句柄被调用,即事件被执行
    49             dispatch_async(mainQueue, ^{
    50                 _btnSendCAPTCHA.enabled = YES;
    51                 [_btnSendCAPTCHA setTitle:@"验证" forState:UIControlStateNormal];
    52                 
    53                 _lblMessage.text = @"使用 GCD 实现倒计时效果";
    54                 _surplusSecond = kSurplusSecond;
    55             });
    56         } else {
    57             _surplusSecond--;
    58             dispatch_async(mainQueue, ^{
    59                 NSString *btnInfo = [NSString stringWithFormat:@"%ld秒", (long)(_surplusSecond + 1)];
    60                 _btnSendCAPTCHA.enabled = NO;
    61                 [_btnSendCAPTCHA setTitle:btnInfo forState:UIControlStateDisabled];
    62             });
    63         }
    64     });
    65     dispatch_source_set_cancel_handler(timer, ^{ //计时器取消处理器;调用 dispatch_source_cancel 时执行
    66         NSLog(@"Cancel Handler");
    67     });
    68     dispatch_resume(timer);  //恢复定时循环计时器;Dispatch Source 创建完后默认状态是挂起的,需要主动恢复,否则事件不会被传递,也不会被执行
    69 }
    70 
    71 - (IBAction)sendCAPTCHA:(id)sender {
    72     _lblMessage.text = [NSString stringWithFormat:@"验证码发送成功,%d秒后可重新发送", kSurplusSecond];
    73     
    74     [self countDown];
    75 }
    76 
    77 @end

    Main.storyboard

     1 <?xml version="1.0" encoding="UTF-8" standalone="no"?>
     2 <document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="7706" systemVersion="14E46" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" initialViewController="vXZ-lx-hvc">
     3     <dependencies>
     4         <deployment identifier="iOS"/>
     5         <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="7703"/>
     6     </dependencies>
     7     <scenes>
     8         <!--View Controller-->
     9         <scene sceneID="ufC-wZ-h7g">
    10             <objects>
    11                 <viewController id="vXZ-lx-hvc" customClass="ViewController" sceneMemberID="viewController">
    12                     <layoutGuides>
    13                         <viewControllerLayoutGuide type="top" id="jyV-Pf-zRb"/>
    14                         <viewControllerLayoutGuide type="bottom" id="2fi-mo-0CV"/>
    15                     </layoutGuides>
    16                     <view key="view" contentMode="scaleToFill" id="kh9-bI-dsS">
    17                         <rect key="frame" x="0.0" y="0.0" width="600" height="600"/>
    18                         <autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMaxY="YES"/>
    19                         <subviews>
    20                             <label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="使用 GCD 实现倒计时效果" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="1Kh-pV-cfz">
    21                                 <rect key="frame" x="200" y="289.5" width="200" height="20.5"/>
    22                                 <fontDescription key="fontDescription" type="system" pointSize="17"/>
    23                                 <color key="textColor" cocoaTouchSystemColor="darkTextColor"/>
    24                                 <nil key="highlightedColor"/>
    25                             </label>
    26                             <button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="center" contentVerticalAlignment="center" buttonType="roundedRect" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="rFe-Xb-ZSc">
    27                                 <rect key="frame" x="240" y="510" width="120" height="50"/>
    28                                 <constraints>
    29                                     <constraint firstAttribute="width" constant="120" id="gVH-aT-gen"/>
    30                                     <constraint firstAttribute="height" constant="50" id="jJP-Vc-fpy"/>
    31                                 </constraints>
    32                                 <state key="normal" title="验证">
    33                                     <color key="titleShadowColor" white="0.5" alpha="1" colorSpace="calibratedWhite"/>
    34                                 </state>
    35                                 <connections>
    36                                     <action selector="sendCAPTCHA:" destination="vXZ-lx-hvc" eventType="touchUpInside" id="I6T-s9-9H6"/>
    37                                 </connections>
    38                             </button>
    39                         </subviews>
    40                         <color key="backgroundColor" white="1" alpha="1" colorSpace="custom" customColorSpace="calibratedWhite"/>
    41                         <constraints>
    42                             <constraint firstAttribute="centerX" secondItem="rFe-Xb-ZSc" secondAttribute="centerX" id="CoE-CP-eDN"/>
    43                             <constraint firstAttribute="centerY" secondItem="1Kh-pV-cfz" secondAttribute="centerY" id="bX4-jS-0xm"/>
    44                             <constraint firstAttribute="centerX" secondItem="1Kh-pV-cfz" secondAttribute="centerX" id="mKH-Zw-Utb"/>
    45                             <constraint firstItem="2fi-mo-0CV" firstAttribute="top" secondItem="rFe-Xb-ZSc" secondAttribute="bottom" constant="40" id="y3Q-IA-qIO"/>
    46                         </constraints>
    47                     </view>
    48                     <connections>
    49                         <outlet property="btnSendCAPTCHA" destination="rFe-Xb-ZSc" id="UFG-TS-ImX"/>
    50                         <outlet property="lblMessage" destination="1Kh-pV-cfz" id="gzx-3T-euc"/>
    51                     </connections>
    52                 </viewController>
    53                 <placeholder placeholderIdentifier="IBFirstResponder" id="x5A-6p-PRh" sceneMemberID="firstResponder"/>
    54             </objects>
    55         </scene>
    56     </scenes>
    57 </document>

    输出结果:

    1 2015-08-31 14:40:02.083 KMCountDown[5102:103949] Event Handler
    2 2015-08-31 14:40:03.087 KMCountDown[5102:103949] Event Handler
    3 2015-08-31 14:40:04.084 KMCountDown[5102:103949] Event Handler
    4 2015-08-31 14:40:05.086 KMCountDown[5102:103971] Event Handler
    5 2015-08-31 14:40:06.085 KMCountDown[5102:103949] Event Handler
    6 2015-08-31 14:40:07.085 KMCountDown[5102:103949] Event Handler
    7 2015-08-31 14:40:07.085 KMCountDown[5102:103949] Cancel Handler
  • 相关阅读:
    html、css、js文件加载顺序及执行情况
    python之路_前端基础之jQuery入门3
    python之路_前端基础之jQuery入门2
    python之路_前端基础之jQuery入门1
    python之路_前端基础之JS5
    python之路_前端基础之JS4
    python之路_前端基础之CSS布局3
    python之路_前端基础之JS(3)
    python之路_前端基础之JS(2)
    python之路_前端基础之JS(1)
  • 原文地址:https://www.cnblogs.com/huangjianwu/p/4773065.html
Copyright © 2011-2022 走看看