iOS 内存管理 内存布局 Copy Tagged Pointer 引用计数存储

我们先看下以下几道题目:

  1. 使用CADisplayLink、NSTimer有什么注意点
  2. 介绍下内存的几大区域
  3. 讲一下你对iOS内存管理的理解
  4. ARC都帮我们做了什么?
  5. weak指针得实现原理

解答:

1. 使用CADisplayLink、NSTimer有什么注意点?

CADisplayLink保证调用频率和屏幕的刷帧一致,60FPS。
CADisplayLink、NSTimer都会对target进行引用,很容易造成循环引用得问题,造成target无法释放。

     self.link = [CADisplayLink displayLinkWithTarget:self selector:@selector(linkTest)];
     [self.link addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
》》》》》》》》》》》》》》》》》》
     self.timer = [NSTimer scheduledTimerWithTimeInterval:1.0f target:self selector:@selector(timerTest) userInfo:nil repeats:YES];

解决方法:
1. timer使用block方法,来对target进行弱引用。
2. 使用一个中间类,弱引用target,然后用消息转发再次转发给target。因为中间类跟target是弱引用,当vc delloc的时候,会正常进行释放,timer也会释放,避免了循环引用得问题。

//JWHelper继承NSObject
#import "JWHelper.h"
@interface JWHelper()
@property (nonatomic, weak) id target;
@end
@implementation JWHelper
+ (instancetype)initHelperTarget:(id)target {
    JWHelper * helper = [JWHelper new];
    helper.target = target;
    return helper;
}
//消息转发给target
- (id)forwardingTargetForSelector:(SEL)aSelector {
    return self.target;
}
@end

>>>>>>>>>>>>>>>>>>>调用>>>>>>>>>>>>>>>>>>>>>
  self.link = [CADisplayLink displayLinkWithTarget:[JWHelper initHelperTarget:self] selector:@selector(linkTest)];
  [self.link addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
   self.timer = [NSTimer scheduledTimerWithTimeInterval:1.0f target: [JWHelper initHelperTarget:self] selector:@selector(timerTest) userInfo:nil repeats:YES];

NSProxy类

在OC中,NSProxy类属于NSObject同级别得基类,NSProxy主要作用就是负责消息转发机制,当转发到的selector之后会直接调用

-(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
-(void)forwardInvocation:(NSInvocation *)anInvocation

两个方法进行消息转发机制,该类转发效率比NSObject效率高,因为他不会从缓存类/superClass内部寻找方法,如果没有找到再进行消息转发。它是直接进行消息转发,所以效率更高。

//JWProxy继承NSProxy
@interface JWProxy()
@property (nonatomic, weak) id target;
@end
@implementation JWProxy
+ (instancetype)initHelperTarget:(id)target {
    JWProxy * proxy = [JWProxy alloc];
    proxy.target = target;
    return proxy;
}
-(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    return [self.target methodSignatureForSelector:aSelector];
}
-(void)forwardInvocation:(NSInvocation *)anInvocation {
    [anInvocation invokeWithTarget:self.target];
}
@end

NSTimer、CADisplayLink是依赖RunLoop实现的,当每次执行一圈RunLoop系统会计算是否满足触发定时器的条件,所以使用RunLoop触发定时器有误差,造成不准时,如果对精度比较高使用GCD定时器。

//GCD创建定时器
@interface JWGCDTimer()
@property (nonatomic, strong) dispatch_source_t timer;
@end

@implementation JWGCDTimer
+ (instancetype)initTimer:(NSTimeInterval)start interval:(NSTimeInterval)interval handler:(void (^)(void))block {
    return [[JWGCDTimer alloc]initTimer:start interval:interval handler:block];
}
- (instancetype)initTimer:(NSTimeInterval)start interval:(NSTimeInterval)interval handler:(void (^)(void))block {
    if (self = [super init]) {
        dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, dispatch_get_main_queue());
        dispatch_source_set_timer(timer, DISPATCH_TIME_NOW, start * NSEC_PER_SEC, interval * NSEC_PER_SEC);
        dispatch_source_set_event_handler(timer, ^{
            if (block) {
                block();
            }
        });
        dispatch_resume(timer);
        self.timer = timer;
    }
    return self;
}

2. 内存布局

iOS程序的内存格局大致分为一下几段:
内存地址从低地址--->高地址:保留区(系统保留)、代码段(__TEXT)、数据段(__DATA)、堆区、栈区、系统内核区域。

  • 代码段: 我们编写得代码,最终变成机器指令存放再代码段、
  • 数据段: 分为:
    1.字符串常量:比如NSString * str = @"123"; @"123"存放在在数据段此区域。
    2.已初始化数据:已初始化的全局变量、静态变量(static int a = 10;)等
    3.未初始化数据:未初始化的全局变量、静态变量等
  • 堆:通过alloc、malloc、calloc等动态分配内存的区域,通过低地址向高地址分配
  • 栈:函数调用开销,比如局部变量得开销。通过高地址向低地址分配,代码越向后地址越低。


    程序内存布局

Tagged Pointer技术

  • 从64bit开始,iOS引入了Tagged Pointer技术来优化NSNumber、NSDate、- NSString等小对象得存储。
  • 在没有使用Tagged Pointer之前,NSNumber等对象需要动态分配内存、维护技术表、NSNumber指针存储得是堆中NSNumber对象的地址值。
  • 使用Tagged Pointer会讲tag+data的形式储存在指针当中。
  • 当(Mac平台)对象地址的最低有效位是1,则该指针为taggedpointer
  • 当(iPhone 开发)对象地址的最高有效位是1,则该指针为taggedpointer
  • 当指针不够存储数据时,才会使用都动态分配内存的方式来存储数据
  • objc_msgSend能识别Tagged Pointer比如NSNumber得intValue方法,直接从指针提取数据,节省了以前的调用开销。
NSString * str = [NSString stringWithFormat:@"abc"];
NSString * str1 = [NSString stringWithFormat:@"abcdefghijklmn"];
NSLog(@"%p=====%p",str,str1);
------------------------------------------------
2018-12-13 14:33:47.574247+0800 Atomic[63696:6388988] 0xa000000006362613=====0x60000003a5c0

如何区别是否是Tagged Pointer?
str:地址的高位0xa 二进制是:1010,有效位是1 属于是Tagged Pointer。
str1:地址的高位0x6 二进制是:0220,有效位是0 不属于是Tagged Pointer。

MRC 引用计数

  • 在iOS种,使用引用计数来管理OC对象的内存。
  • 一个新创建得OC对象引用计数默认是1,当引用计数为0的时候OC对象就会销毁,释放其占用的内存空间
  • 调用retain会让OC对象的引用计数+1,调用release会-1
  • 当调用alloc、new、copy、mutableCopy方法返回一个对象,在不需要得时候都需要release或者autoRelease释放他
//MRC下setter方法下手动内存管理,先释放旧值,再保存新值。
//@property(nonatomic,retain) id obj 原理
- (void)setDog:(JWDog *)dog {
    if (_dog != dog) {
        [_dog release];
        _dog = [dog retain];
    }
}

Copy

  • copy:产生不可变副本,浅拷贝,指针拷贝,没有产生新的对象,产生一个新指针指向该内容区域,相当于retain
  • mutableCopy:产生可变副本,深拷贝,内容拷贝,复制一份全新的内容

[不可变str copy] 浅拷贝
[不可变str mutableCopy] 深拷贝
[可变str copy] 深拷贝
[可变str mutableCopy] 深拷贝


深浅拷贝主要是看是否拷贝源是否可变。
浅拷贝 相当于执行了一次retain操作,并没有产生新对象,引用计数会+1
深拷贝 相当于创建了一个新对象,新对象引用计数初始为1,原对象并没有+1。

拷贝分析

引用计数的存储

从arm64位之后,引用计数是存储在isa指针中的。isa指针进行了优化,使用union(共用体)来存储指针。

isa指针

其中 uintptr_t has_sideTable_rc:1 、 extra_rc:19就是来存储引用计数的(实际值是存储引用计数-1)。
从isa指针中的19位二进制位来存储引用计数,当存储超过上限的时候会让
has_sideTable_rc值为1,然后将存储在一个SideTable的类的属性中。
objc4/NSObject.mm

struct SideTable {
    spinlock_t slock;  //自旋锁
    RefcountMap refcnts; //存储的哈希表
    weak_table_t weak_table;  //弱引用的哈希表
...
};

查看引用计数的源码也可以看看出


引用计数的获取

weak指针的实现原理

当一个__weak指向一个对象的时候,会将此对象的地址值作为key,放入全局的SideTable中的一个叫weak_table的哈希表中存储,当该对象该释放的时候,会根据isa指针中的weakly_referenced属性来检查是否存在弱引用,存在的话,系统根据weak_table表中将该对象的地址值作为key&一个mask值来找到存储得位置并且删除,外部将此对象置为nil,并且释放该对象,回收内存。

ARC是由LLVM编译器(大括号后面自动调用release)和Runtime(动态销毁对象)协同产生的一种技术。

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