美文网首页
线程保活

线程保活

作者: Aliv丶Zz | 来源:发表于2020-09-12 13:20 被阅读0次

在实际开发中,我们可能很多操作需要放在子线程中操作,可能会重复的创建线程。这个时候我们就需要创建一个线程并不让其销毁,然后将需要放在子线程中的操作放在里面进行。

实例:

.h

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

typedef void(^ZLXThreadTask)(void);

@interface ZLXPerennialThread : NSObject

//-(void)run;

/**
执行任务
 */

-(void)executeTask:(ZLXThreadTask) task;

/**
停止线程
 */
-(void)stop;

@end

NS_ASSUME_NONNULL_END

.m

#import "ZLXPerennialThread.h"
@interface ZLXThread : NSThread
@end
@implementation ZLXThread
-(void)dealloc{
    NSLog(@"-- %s --",__func__);
}
@end


@interface ZLXPerennialThread()
@property (nonatomic, strong) ZLXThread *interThread;
@property (nonatomic, assign, getter=isStop) BOOL stop;


@end

@implementation ZLXPerennialThread

- (instancetype)init
{
    self = [super init];
    if (self) {
        __weak typeof(self) weakSelf = self;

        self.interThread = [[ZLXThread alloc] initWithBlock:^{
#if 0
            // OC runloop
            [[NSRunLoop currentRunLoop] addPort:[[NSPort alloc]init] forMode:NSDefaultRunLoopMode];
                   while (weakSelf && !weakSelf.isStop) {
                       [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
                   }
#else
            // C runloop   {0}:初始化结构体,如果不初始化,可能数据错乱
            CFRunLoopSourceContext context = {0};
            CFRunLoopSourceRef source = CFRunLoopSourceCreate(kCFAllocatorDefault, 0, &context);
            CFRunLoopAddSource(CFRunLoopGetCurrent(), source, kCFRunLoopDefaultMode);
        #if 0
            CFRunLoopRunInMode(kCFRunLoopDefaultMode, 1.0e10, false);
        #else
            while (weakSelf && !weakSelf.isStop) {
                 /**
                  1.0e10: CFRunLoopRun()  源码中 的默认值
                   true:returnAfterSourceHandled执行完后,退出当前runloop
                  */
                 CFRunLoopRunInMode(kCFRunLoopDefaultMode, 1.0e10, true);
             }
        #endif
#endif
        }];
        [self.interThread start];

    }
    return self;
}

//-(void)run{
//    if (!self.interThread) return;
//    
//    [self.interThread start];
//}

-(void)executeTask:(ZLXThreadTask)task{
    
    if (!self.interThread || !task) return;
    
    if (!self.interThread.isExecuting) {
        [self.interThread start];
    }
    
    [self performSelector:@selector(__executeTask:) onThread:self.interThread withObject:task waitUntilDone:YES];

}

-(void)stop{
    if (!self.interThread) return;
    
    [self performSelector:@selector(__stop) onThread:self.interThread withObject:nil waitUntilDone:YES];

}

#pragma -makr privet methods

-(void)__stop{
    self.stop = YES;
    CFRunLoopStop(CFRunLoopGetCurrent());
    self.interThread = nil;

}

-(void)__executeTask:(ZLXThreadTask)task{
    task();
}

-(void)dealloc{
    [self stop];
    NSLog(@"ZLXPerennialThread dealloc");
}
@end

相关文章

  • RunLoop -- 在实际开发中的应用

    1、控制线程生命周期<线程保活> 线程保活 2、解决NSTimer在滑动时失效的问题 当scrollView滑动的...

  • iOS底层原理——浅谈RunLoop

    RunLoop应用:线程保活 线程保活、控制销毁 iOS-浅谈RunLoop8iOS底层原理总结 - RunLoo...

  • iOS NSThread 保活线程代码封装

    iOS NSThread 保活线程代码封装

  • iOS Runloop的理解与使用

    Runloop的概念 Runloop的存在主要就是为了线程保活,线程保活是为了线程能够及时的处理事件,不会在其执行...

  • 线程保活

    在实际开发中,我们可能很多操作需要放在子线程中操作,可能会重复的创建线程。这个时候我们就需要创建一个线程并不让其销...

  • 线程保活

    线程保活是在多线程中进行耗时操作常用的功能: 常规开启方式,会出现内存泄漏 通过 [runloop run]直接...

  • 线程保活

    #import "ViewController.h" #import "WZJthread.h" @interfa...

  • 线程保活

    线程保活 当子线程中的任务执行完毕后,线程就被立刻销毁了。如果程序中,需要经常在子线程中执行任务,频繁的创建和销毁...

  • 线程保活

    ViewController.h ViewController.m

  • iOS底层探索 --- RunLoop(实战)

    日常开发中我们常用的RunLoop场景有: 线程保活 Timer相关 APP卡顿检测 线程保活首先我们应该达成的共...

网友评论

      本文标题:线程保活

      本文链接:https://www.haomeiwen.com/subject/bcdwektx.html