这些 iOS 冷知识,你知道吗?

单例对象的内存管理

问题背景

在解决 App 防止抓包问题的时候,有一种常见的解决方案就是:检测是否存在代理服务器。其实现为:

+ (BOOL)getProxyStatus {

CFDictionaryRefdicRef =CFNetworkCopySystemProxySettings();

constCFStringRefproxyCFstr=CFDictionaryGetValue(dicRef, (constvoid*)kCFNetworkProxiesHTTPProxy);

CFRelease(dicRef);

NSString*proxy = (__bridgeNSString*)(proxyCFstr);

if(proxy) {

returnYES;

}

returnNO;

}

在我前面的一篇文章《iOS 内存泄漏场景与解决方案》中,有提到非 OC 对象在使用完毕后,需要我们手动释放。

那么上面这段代码中,在执行 CFRelease(dicRef); 之后,dicRef 是不是应该就被释放了呢?

问题探讨

让我们来写一段测试代码试试看:

CFDictionaryRef dicRef = CFNetworkCopySystemProxySettings();

NSLog(@"%ld, %p", CFGetRetainCount(dicRef), dicRef);

CFRelease(dicRef);

NSLog(@"%ld, %p", CFGetRetainCount(dicRef), dicRef);

CFRelease(dicRef);

NSLog(@"%ld, %p", CFGetRetainCount(dicRef), dicRef);

打印结果为:

2,0x6000004b9720

1,0x6000004b9720

(lldb)

程序在运行到第三次 NSLog 的时候才崩溃,说明对 dicRef 对象 release 两次才能将他彻底释放。

这很奇怪,按照以往的经验,第一次打印 dicRef 的引用计数值不应该是 1 才对吗?

修改一下代码,继续测试:

CFDictionaryRef dicRef = CFNetworkCopySystemProxySettings();

CFRelease(dicRef);

CFRelease(dicRef);

NSLog(@"%p", CFNetworkCopySystemProxySettings());

这次运行到最后一行代码的时候,居然还是崩溃了。连 CFNetworkCopySystemProxySettings() 对象都直接从内存里被销毁了?难道 dicRef 没有重新创建对象,而是指向了真正的地址?

为了验证猜想,我们定义两份 dicRef 对象,并打印出他们的地址和引用计数。

CFDictionaryRef dicRef = CFNetworkCopySystemProxySettings();

NSLog(@"%p, %ld,", dicRef, CFGetRetainCount(dicRef));

CFDictionaryRef dicRef1 = CFNetworkCopySystemProxySettings();

NSLog(@"%p, %p, %ld, %ld", dicRef, dicRef1, CFGetRetainCount(dicRef), CFGetRetainCount(dicRef1));

打印结果为:

0x600003bd2040,2,

0x600003bd2040,0x600003bd2040,3,3

果然如此。dicRef 和 dicRef1 的地址是一样的,而且第二次打印时,在没有对 dicRef 对象执行任何操作的情况下,它的引用计数居然又加了 1。

那么我们可以大胆猜测:

实际上,每次调用 CFNetworkCopySystemProxySettings() 返回的地址一直是同一个,未调用时它的引用计数就为 1,而且每调用一次,引用计数都会加 1。

如此看来,CFNetworkCopySystemProxySettings() 返回的对象在引用计数上的表现和其它系统单例十分相似,比如 [UIApplication sharedApplication]、[UIPasteboard generalPasteboard]、[NSNotificationCenter defaultCenter] 等。

单例对象一旦建立,对象指针会保存在静态区,单例对象在堆中分配的内存空间,只在应用程序终止后才会被释放。

因此对于这类单例对象,调用一次就需要释放一次(ARC 下 OC 对象无需手动释放),保持它的引用计数为 1(而不是 0),保证其不被系统回收,下次调用时,依然能正常访问。

block 属性用什么修饰

问题背景

这个问题来源于一道司空见惯的面试题:

iOS 种 block 属性用什么修饰?(copy 还是 strong?)

Stack Overflow 上也有相关的问题:Cocoa blocks as strong pointers vs copy。

问题探讨

先来回顾一些概念。

iOS 内存分区为:栈区、堆区、全局区、常量区、代码区(地址从高到低)。常见的 block 有三种:

• NSGlobalBlock:存在全局区的 block;

• NSStackBlock:存在栈区的 block;

• NSMallocBlock:存在堆区的 block。

block 有自动捕获变量的特性。当 block 内部没有引入外部变量的时候,不管它用什么类型修饰,block 都会存在全局区,但如果引入了外部变量呢?

这个问题要在 ARC 和 MRC 两种环境下讨论。

Xcode 中设置 MRC 的开关:

1、全局设置:TARGETS → Build Settings → Apple Clang - Language - Objective-C → Objective-C Automatic Reference Counting 设为 No;(ARC 对应的是 Yes)

2、局部设置:TARGETS → Build Phases → Compile Sources → 找到需要设置的文件 → 在对应的 Compiler Flags 中设置 -fno-objc-arc。(ARC 对应的是 -fobjc-arc)

针对这个问题,网上有一种答案:

MRC 环境下,只能用 copy 修饰。使用 copy 修饰,会将栈区的 block 拷贝到堆区,但 strong 不行;

ARC 环境下,用 copy 和 strong 都可以。

看似没什么问题,于是我在 MRC 环境执行了如下代码:

// 分别用 copy 和 strong 修饰 block 属性

@property(nonatomic,copy)void(^copyBlock)(void);

@property(nonatomic,strong)void(^strongBlock)(void);

intx =0;

// 打印 normalBlock 所在的内存地址

void(^normalBlock)(void) = ^{

NSLog(@"%d", x);

};

NSLog(@"normalBlock: %@", normalBlock);

// 打印 copyBlock 所在的内存地址

self.copyBlock= ^(void) {

NSLog(@"%d", x);

};

NSLog(@"copyBlock: %@",self.copyBlock);

// 打印 strongBlock 所在的内存地址

self.strongBlock= ^(void) {

NSLog(@"%d", x);

};

NSLog(@"strongBlock: %@",self.strongBlock);

打印结果为:

normalBlock:<__NSStackBlock__:0x7ffeee29b138>

copyBlock:<__NSMallocBlock__:0x6000021ac360>

strongBlock:<__NSMallocBlock__:0x600002198240>

从 normalBlock 的位置,我们可以看出,默认是存在栈区的,但是很奇怪的是,为什么 strongBlock 位于堆区?难道 MRC 时期用 strong 修饰就是可以的?

其实不然,要知道 MRC 时期,只有 assign、retain 和 copy 修饰符,strong 和 weak 是 ARC 时期才引入的。

strong 在 MRC 中对应的是 retain,我们来看一下在 MRC 下用这两个属性修饰 block 的区别。

// MRC 下分别用 copy 和 retain 修饰 block 属性

@property(nonatomic,copy)void(^copyBlock)(void);

@property(nonatomic, retain)void(^retainBlock)(void);

// 打印 copyBlock 所在的内存地址

intx =0;

self.copyBlock= ^(void) {

NSLog(@"%d", x);

};

NSLog(@"copyBlock: %@",self.copyBlock);

// 打印 retainBlock 所在的内存地址

self.retainBlock= ^(void) {

NSLog(@"%d", x);

};

NSLog(@"retainBlock: %@",self.retainBlock);

打印结果为:

copyBlock:<__NSMallocBlock__:0x6000038f96b0>

retainBlock:<__NSStackBlock__:0x7ffeed0a90e0>

我们可以看到用 copy 修饰的 block 存在堆区,而 retain 修饰的 block 存在栈区。

那么修饰符的作用在哪里,为什么会出现不同的结果,我们通过反汇编来探究一下。

把断点打在 self.copyBlock 的声明函数这一行(在上述引用代码的第7行,不是 block 内部)。然后开启 Debug → Debug Workflow → Always show Disassembly 查看汇编代码,点击 Step into。

在 callq 指令中可以看到声明的 copyBlock 属性具有 copy 的特性。

然后断点打在 self.retainBlock 的声明函数这一行,再进入查看,可以注意到 retainBlock 不具有copy 的特性。

再在 ARC 下试一试。把断点打在 self.strongBlock 的声明函数这一行,进入查看,可以发现,用 strong 修饰的属性,也具有 copy 的特性。

这也就很好解释了为什么 MRC 下用 retain 修饰的属性位于栈区,而用 copy、strong 修饰的属性存在堆区。

MRC 下,在定义 block 属性时,使用 copy 是为了把 block 从栈区拷贝到堆区,因为栈区中的变量出了作用域之后就会被销毁,无法在全局使用,而把栈区的属性拷贝到堆区中全局共享,就不会被销毁了。

ARC 下,不需要使用 copy 修饰,因为 ARC 下的 block 属性本来就在堆区。

那为什么开发者基本上都只用 copy 呢?

这是 MRC 的历史遗留问题,上面也说到了,strong 是 ARC 时期引入的,开发者早已习惯了用 copy 来修饰 block 罢了。

最后再补充一个小知识点。

// ARC 下定义 normalBlock 后再打印其所在的内存地址

void(^normalBlock)(void) = ^{

NSLog(@"%d", x);

};

NSLog(@"normalBlock: %@", normalBlock);

// 直接打印某个 block 的内存地址

NSLog(@"block: %@", ^{

NSLog(@"%d", x);

});

复制代码

打印结果为:

normalBlock:<__NSMallocBlock__:0x600001ebe670>

block:<__NSStackBlock__:0x7ffee8752110>

复制代码

block 的实现是相同的,为什么一个在堆区,一个在栈区?

这个现象叫做运算符重载。定义 normalBlock 的时候 = 实际上执行了一次 copy,为了管理 normalBlock 的内存,它被转移到了堆区。


原文地址:这些 iOS 冷知识,你知道吗? - CocoaChina_一站式开发者成长社区

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,245评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,749评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,960评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,575评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,668评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,670评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,664评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,422评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,864评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,178评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,340评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,015评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,646评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,265评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,494评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,261评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,206评论 2 352