ios多线程操作(七)—— GCD延迟操作与一次性代码

13 篇文章 0 订阅
使用GCD函数可以进行延时操作,该函数为
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
    });


  现在我们来分解一下参数
dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC)) :
NSEC_PER_SEC在头文件中的定义如下:
#define NSEC_PER_SEC 1000000000ull /* nanoseconds per second */
该参数表示从现在开始经过多少纳秒
dispatch_get_main_queue():表示主队列
^{
    }:表示一个block任务。
   我们可以来测试一下经过多少纳秒之后,由主队列调度任务是异步执行还是同步执行,代码如下:
 
 //  when 时间 从现在开始经过多少纳秒
    dispatch_time_t when = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.0 * NSEC_PER_SEC));
   
    void (^task)() = ^ {
        // 延迟操作执行的代码
        NSLog(@"%@", [NSThread currentThread]);
    }; 
    // 经过多少纳秒,由主队列调度任务异步执行
    dispatch_after(when, dispatch_get_main_queue(), task);
    // 先执行就是异步,后执行就是同步
    NSLog(@"come here");

执行结果如下:

由此可见主队列中调度任务是异步执行的
再将执行队列改为全局队列和串行队列,得到的结果完全是一样的,由此可知该函数执行的是异步操作。

     GCD中有个函数能够保证某段代码在程序运行过程中只被执行1次!该函数如下:
static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
       
    })

dispatch_once_t在头文件中得定义如下:
typedef long dispatch_once_t;
由此可知该类型是个long类型。当onceToken等于0时就会执行block代码。dispatch_once是线程安全的,只要涉及到线程安全就会涉及到锁,dispatch_once内部也有一把锁,性能比互斥锁高!
利用该函数我们可以来写一个单例模式
     单例模式可以保证在程序运行过程,一个类只有一个实例且该实例易于供外界访问,从而方便控制实例个数,并节约系统资源,当应用程序需要共享一份资源时就可以用单例模式来实现。单例模式分ARC与MRC两种情况,我们可以用宏判断是否为ARC环境

#if __has_feature(objc_arc)
// ARC
#else
// MRC
#endif

ARC环境下简单地单例模式:
@implementation SoundTools
// 定义一个静态成员,保存唯一的实例
static id instance;

// 保证对象只被分配一次内存空间,通过dispatch_once能够保证单例的分配和初始化是线程安全的
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        instance = [super allocWithZone:zone];
    });
    return instance;
}

// 保证对象只被初始化一次
+ (instancetype)sharedSoundTools {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        instance = [[self alloc] init];
    });
    return instance;
}

- (id)copyWithZone:(NSZone *)zone {
    return instance;
}
@end

测试代码如下:
- (void)viewDidLoad {
    [super viewDidLoad];
    
    SoundTools *s1 = [SoundTools sharedSoundTools];
    NSLog(@"%p", s1);
}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
    SoundTools *s2 = [SoundTools sharedSoundTools];
    
    NSLog(@"%p", s2);
}


两个方法打印出来的地址完全一样!

在MRC环境下有如下代码:
// 定义一个静态成员,保存唯一的实例
static id instance;

// 保证对象只被分配一次内存空间,通过dispatch_once能够保证单例的分配和初始化是线程安全的
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        instance = [super allocWithZone:zone];
    });
    return instance;
}

// 保证对象只被初始化一次
+ (instancetype)sharedSoundTools {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        instance = [[self alloc] init];
    });
    return instance;
}

- (id)copyWithZone:(NSZone *)zone {
    return instance;
}

#pragma mark - MRC内存管理方法
/**
 因为单例的对象是保存在静态区的,因此需要重写 内存管理方法,取消默认的引用计数操作!
 */
// 默认会将引用计数-1
- (oneway void)release {
    // 什么也不做,跟highlight类似
}

// 默认引用计数+1,同时返回一个对象
- (instancetype)retain {
    return instance;
}

// 默认添加自动释放标记,延迟释放!
- (instancetype)autorelease {
    return instance;
}

// 返回有多少个对象对当前对象引用的数值
- (NSUInteger)retainCount {
    // 出处:limits.h 会根据CPU的架构自行调整整数的长度
    return ULONG_MAX;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
引用\[1\]:根据官方文档,从3.3.1版本开始,SQLite支持多线程并发访问数据库,但要求设置为SQLITE_THREADSAFE模式。每个数据库连接只能在一个线程中使用,不可多个线程共用一个数据库连接。\[1\]引用\[2\]:SQLite默认情况下只支持单个数据库连接操作,多连接并发操作时会出问题,需要开启wal模式才能进行多项城并发读操作,写操作仍不能并发。\[2\]根据这些信息,可以得出以下结论:在SQLite中,多线程操作数据库有两种解决方案。第一种是多线程读和写,只使用一个数据库连接,这种情况下由于使用的是同一个数据库连接,SQLite内部有锁机制,不会出现异常。但是这种多线程操作并不是真正的并发操作,由于锁机制的存在,仍然是阻塞的。第二种是多线程读和单线程写,每个线程使用独立的数据库连接,并开启wal模式。这种情况下可以实现多线程并发读操作,但写操作仍然只能单线程进行。如果多个线程同时写入(每个线程使用独立的数据库连接),就会出现问题。\[2\]在实际测试中,方案一的用时是方案二的两倍多。\[2\]关于方案一的具体实现,有两种方式:一是开启一个数据库连接永不关闭,二是自己编写一个管理类来实现。\[2\]要开启wal模式,可以使用SQLiteDatabase.enableWriteAheadLogging()方法。\[2\]在多线程操作SQLite时,可能会遇到数据库被锁定的问题,这可能是由于子线程执行的query函数有问题导致的。\[3\] #### 引用[.reference_title] - *1* [SQLite的多线程应用与iOS开发](https://blog.csdn.net/pcliuguangtao/article/details/8123473)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [SQLite多线程并发操作](https://blog.csdn.net/a987687115/article/details/84566282)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [Android SQLite多线程读写和线程同步源码分析](https://blog.csdn.net/z979451341/article/details/79593551)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值