博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
iOS Tagged Pointer (源码阅读必备知识)
阅读量:6073 次
发布时间:2019-06-20

本文共 2310 字,大约阅读时间需要 7 分钟。

Tagged Pointer 介绍

苹果对于Tagged Pointer特点的介绍:

  1. Tagged Pointer专门用来存储小的对象,例如NSNumber和NSDate
  2. Tagged Pointer指针的值不再是地址了,而是真正的值。所以,实际上它不再是一个对象了,它只是一个披着对象皮的普通变量而已。所以,它的内存并不存储在堆中,也不需要malloc和free。
  3. 在内存读取上有着3倍的效率,创建时比以前快106倍。

为什么要引入Tagged Pointer

iPhone5s 采用64位处理器。

对于64位程序,我们的数据类型的长度是跟CPU的长度有关的。

这样就导致了 一些对象占用的内存会翻倍。

同时 维护程序中的对象需要 分配内存,维护引用计数,管理生命周期,使用对象给程序的运行增加了负担。

Tagged Pointer

为了改进上面提到的内存占用和效率问题,苹果提出了Tagged Pointer对象。由于NSNumber、NSDate一类的变量本身的值需要占用的内存大小常常不需要8个字节,拿整数来说,4个字节所能表示的有符号整数就可以达到20多亿(注:2^31=2147483648,另外1位作为符号位),对于绝大多数情况都是可以处理的。

我们可以将一个对象的指针拆成两部分,一部分直接保存数据,另一部分作为特殊标记,表示这是一个特别的指针,不指向任何一个地址。所以,引入了Tagged Pointer对象之后,64位CPU下NSNumber的内存图变成了以下这样:

Tagged Pointer

测试

#import int main(int argc, const char * argv[]) {    @autoreleasepool {        // insert code here...        NSNumber *number1 = @1;        NSNumber *number2 = @2;        NSNumber *number3 = @3;        NSNumber *numberFFFF = @(0xFFFF);        NSNumber *numberLager = @(MAXFLOAT);        NSLog(@"number1 pointer is %p", number1);        NSLog(@"number2 pointer is %p", number2);        NSLog(@"number3 pointer is %p", number3);        NSLog(@"numberLager pointer is %p", numberLager);        /*         2017-03-10 12:07:50.731726 TaggedPoint[1690:50438] number1 pointer is 0x127         2017-03-10 12:07:50.731992 TaggedPoint[1690:50438] number2 pointer is 0x227         2017-03-10 12:07:50.732011 TaggedPoint[1690:50438] number3 pointer is 0x327         2017-03-10 12:07:50.732043 TaggedPoint[1690:50438] numberLager pointer is 0x1002006a0         */    }    return 0;}复制代码

以 0x127 为例 去掉 tag27(假设27为标记) 0x1 就是number 的值。

0x227
0x327
都有这种规律

numberLager 存储的值为MAXFloat 显然超过了tagged pointer 可以存储的范围。

所以打印的地址是单纯的指针地址,指向存储numberLager的内存地址。

对于isa指针的影响

因为tagged pointer 不是一个真正的对象,如果使用isa指针在编译时会报错。

如图:

提示我们改为object_getClass()
object_getClass()中做了相应的处理

由于object_getClass()没有对应的实现,只能从其他地方窥探一二

objc-weak.mm

weak_read_no_lock(weak_table_t *weak_table, id *referrer_id) {    objc_object **referrer = (objc_object **)referrer_id;    objc_object *referent = *referrer;    if (referent->isTaggedPointer()) return (id)referent;    //...}复制代码
inline bool objc_object::isTaggedPointer() {#if SUPPORT_TAGGED_POINTERS    return ((uintptr_t)this & TAG_MASK);#else    return false;#endif}复制代码

这里取对象的值做了一些判断

如果是tagged pointer , 对象的值就是指针
如果非tagged pointer , 对象的值是指针指向的内存区域中的值

转载地址:http://ydigx.baihongyu.com/

你可能感兴趣的文章
Spring Cloud云服务- HongHu云架构common-service代码结构分析
查看>>
【BZOJ】1585: [Usaco2009 Mar]Earthquake Damage 2 地震伤害
查看>>
Java开发5年只会curd,被新来开发一年的小子说成是混吃等死...
查看>>
CentOS 7 Docker
查看>>
Telegraf+Influxdb+Grafana构建监控平台
查看>>
死磕Tomcat系列(1)——整体架构
查看>>
Spring AOP
查看>>
The following users do not have email address specified
查看>>
Docker Data Center系列(一)- 快速搭建云原生架构的实践环境
查看>>
ESLint 禁止检验
查看>>
用了那么多年MySQL不知道Explain?怪不得性能那么差!
查看>>
java 字符流 字节流
查看>>
mysql实战14 | count(*)这么慢,我该怎么办?
查看>>
wireshark的ubuntu更新ppa源
查看>>
AngularJs 学习 (二)
查看>>
0302感想
查看>>
实验四 主存空间的分配和回收
查看>>
解决eclipse之ADT与SDK版本不一致问题
查看>>
jQuery 属性操作
查看>>
小甜点,RecyclerView 之 ItemDecoration 讲解及高级特性实践
查看>>