__block int timeout=60;
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(), ^{
[self.validBtn setTitle:@"获取短信验证码" forState:UIControlStateNormal];
//self.validBtn.backgroundColor=RGB(120, 0, 103);
[self.validBtn setBackgroundImage:[UIImage imageNamed:@"获取验证码"] forState:UIControlStateNormal];
self.validBtn.userInteractionEnabled = YES;
});
}else{
int seconds = timeout;
NSString *strTime = [NSString stringWithFormat:@"%.2d", seconds];
dispatch_async(dispatch_get_main_queue(), ^{
[self.validBtn setTitle:[NSString stringWithFormat:@"%@秒",strTime] forState:UIControlStateNormal];
//self.validBtn.backgroundColor=RGB(207, 207, 207);
[self.validBtn setBackgroundImage:[UIImage imageNamed:@"获取验证码-不可点击"] forState:UIControlStateNormal];
self.validBtn.userInteractionEnabled = NO;
});
timeout--;
}
});
dispatch_resume(_timer);
現在のページを離れるときにスレッドを終了し、カウントダウンを 60 に初期化します
タイマーは Runloop で実行する必要があります
バックグラウンドに移行した後、バックグラウンドで実行できるプログラムとして宣言されていない場合、NSTimer は消えないはずです。
私だったら:
1. バックグラウンドに入る前にタイマーを停止し、appdelegate の applicationDidEnterBackground に時間を記録し、フォアグラウンドに入るときの時間を比較し、差分を計算し、検証コード値の表示を更新します。そしてタイマーを開始します。
2. バックグラウンドに入る前にタイマーを停止するか、backgroundTask を使用してバックグラウンドで数秒ごとに確認コード番号を減算し、フロントデスクに戻ってからタイマーを開始します。 (これはとても退屈です...)
3. アプリをバックグラウンドで実行できるプログラムとして宣言します。
私なら最初の選択肢を選びます。 。 。
私の計画は次のとおりです:
録画開始時刻。
インターフェース上の表示を毎秒更新します。
当前时间 - 开始时间
を使用してインターフェースに表示される秒数を取得しますを使用しています。バックグラウンドに入ってからフォアグラウンドに戻るときの状況を追加で処理する必要はありません。もちろん、ここで処理するだけで問題ありません。 。
GCD Timer
次の解決策を試すことができます。
リーリーGreedTimer はあなたが指摘した問題を解決できます。 GRTimer を使用する場合は、
に設定してくださいtimeInBackground
をYES
上記の方法はいずれも機能しません。簡単で実行可能な方法は見つかりませんでした。
http://www.jianshu.com/p/8e61...