一:汇编分析String底层
面试题
1个String变量占用多少内存?
下面2个String变量,底层存储有什么不同?
var str1 = "0123456789"
var str2 = "0123456789ABCDEF"
如果对String进行拼接操作,String变量的存储会发生什么变化?
str1.append("ABCDE")
str1.append("F")
str2.append("G")
题目1: 个String变量占用多少内存?
import Foundation
var str1 = "0123456789"
-
打开debug汇编:debug->debug Workflow->Always show Disassembly
1.0x100003f6a <+26>: movq %rax, 0x409f(%rip) ; Study_Share.str1 : Swift.String:
占用8个字节
-
0x100003f71 <+33>: movq %rdx, 0x40a0(%rip) ; Study_Share.str1 : Swift.String + 8 0x100003f71 <+33>: movq %rdx, 0x40a0(%rip) ; Study_Share.str1 : Swift.String + 8:
占用高地址8个字节
所以总共占用16
字节
- 另一种验证方式
print(MemoryLayout.stride(ofValue: str1)) 输出
16
题目2: 下面2个String变量,底层存储有什么不同?
var str1 = "0123456789"
var str2 = "0123456789ABCDEF"
我们来继续分析str1内存地址:
地址a+地址b=str1的内存地址 0x100008038
我们用x/2xg打印该地址
0x100008038
会发现很有趣的问题ASCII码表
`(lldb) x/2xg 0x100008038
0x100008038: 0x3736353433323130 0xea00000000003938`
发现有个特点: 30表示为0,
我们发现
0x3736353433323130 0xea00000000003938
地址从小端模式30到39就已经表示了"0123456789", 那0xea
其中a
表示自负串长度,如a等于10
。那问题来了a
这个地址的最大值多少呢?是的,你猜想的是对的,最大值是15
,即自符串最大长度为15。字符串的值直接存在用地址表示,这个就很类似OC的tagger pointer
现在我们来试试你的猜想,试试字符串长度大于15的情况
var str1 = "0123456789ABCDEF"
此时str1的真实地址就是rip+rdi
(lldb) x/2xg 0x100008038
0x100008038: 0xd000000000000010 0x8000000100003f60
说明0123456789ABCDEF
就不再存储在0xd000000000000010 0x8000000100003f60
上面了。
接下来我们汇编分析下var str1 = "0123456789ABCDEF"
的流程及怎么存储的
我们给callq
方法打上断点,然后进入到这个函数:
方式1: 按住control键 然后点击如图鼠标处图标
方式2:
打印台输入si即可
(结束finish)如图会进入
-
Swift的 String.init初始化器
然后会执行
cmpq $0xf, %rsi
rsi
表示字符串长度
然后cmpq
作用的用来比较大小的,cmpq $0xf, %rsi
就是rsi
比较0xf
(长度15
)。
15是分界线,会走到不同的流程
0x100008038: 0xd000000000000010 0x8000000100003f60
字符串的真实地址
:0x7fffffffffffffe0 - 0x8000000100003f60
字符串的真实地址
: 0x0000000100003f60 + 0x20 0x8换成0x0
0x8000000100003f60 - 0x7fffffffffffffe0 = 0x100003F80
(lldb) x 0x100003F80
0x100003f80: 30 31 32 33 34 35 36 37 38 39 41 42 43 44 45 46 0123456789ABCDEF
0x100003f90: 00 0a 00 20 00 00 00 00 18 fe ff ff 03 00 00 00 ... ............
真实地址0x100003F80
发现他就存储着 0123456789ABCDEF
的值
那0xd000000000000010
又有啥作用呢 其实这个10
表示的是自负串长度
- 扩展: 那
0123456789ABCDEF
具体存储在什么局呢
其实可以直接看mach-o文件的,我们直接使用MachOView软件查看
如图所示:存储在常量区
`//字符串长度<=0xF, 字符串内存直接存在str1变量的内存中`
var str1 = "0123456789"
`//字符串长度>0xF, 字符串内容存放在__TEXT.cstring中(常量区)常量区是没法再更改的`
var str2 = "0123456789ABCDEFR"
题目三:如果对String进行拼接操作,String变量的存储会发生什么变化?
str1.append("ABCDE")
str1.append("F")
str2.append("G")
结论:
1. 字符串长度<=0xF, 字符串内存直接存在str1变量的内存中
var str1 = "0123456789"
2. 直接字面量创建的 字符串长度>0xF, 字符串内容存放在__TEXT.cstring中(常量区) 常量区是没法再更改的
var str2 = "0123456789ABCDEFR"
3. 由于字符串长度<=0xF 15位,所以字符串内容依然存放在str1变量的内存中
str1.append("F")
4. 再次append之后字符串长度>0xF,这个时候内存地址已无法再存放,所以会开辟堆空间
str1. append("GHH")
注:在这里我们就汇编分析证明字符串存储在堆上的情况了,因为发现不同系统版本卡的流程有较大区别,很繁杂
二:关于Array的思考
官方定义的数组是结构体(值类型):
public struct Array<Element>
- 1个Array变量占用多少内存?
构体的内存占用大小是把存放到结构体中的变量占用内存大小加起来(字节对齐
)。
示例代码:
struct Point {
var x = 0, y = 0
}
var p = Point()
print(MemoryLayout.stride(ofValue: p))
// 输出:16
var arr = [1, 2, 3, 4]
print(MemoryLayout.stride(ofValue: arr))
// 输出:8
结构体一共占用16个字节内存。
数组也是结构体,占用内存大小的计算方法是否和上面的示例代码一致呢?
很遗憾,只占用8个字节内存,且是Int
类型占用的内存大小。那么数组里面的内容是存放在哪里呢?
- 数组中的数据存放在哪里?
通过汇编分析可以知道,数组中数据是存放在堆空间的,数组变量内存存放着堆空间的数组对象地址。
示例代码:
var arr = [1, 2, 3, 4]
print(Mems.memStr(ofRef: arr))
/*
输出:
0x00007fff8e5f54d8
0x0000000200000002
0x0000000000000004
0x0000000000000008
0x0000000000000001
0x0000000000000002
0x0000000000000003
0x0000000000000004
*/
通过内存布局看到,数组内容需要跳过前面的32个字节。那么前面的字节分别存放着什么东西呢?
第一段8个字节:存放着数组相关引用类型信息内存地址
第二段8个字节:数组的引用计数
第三段8个字节:数组的元素个数
第四段8个字节:数组的容量
后面依次存放着数组的元素
数组的容量会自动扩容至元素个数的两倍,且是8的倍数。
三:总结
平时我都知道他们是值类型,但是通过汇编窥探发现,String 也有存放在堆空间的情况, Array 就是放在堆空间的,这不是和 Swift 说的 值类型冲突了吗? 并不是这样,底层实现是苹果官方的做法,他把数组设计成引用类型,只限于在底层的实现 , 但是对于我们使用着 String 和 Array 的人来说, String 和 Array 行为表现官方定义它就是表现为值类型,我们开发者看到 String 和 Array ,就知道是值类型,把它当做值类型来用。
四:复习OC-NSString
面试题
以下代码会有什么问题?
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
dispatch_queue_t queue = dispatch_queue_create("com.cjl.cn", DISPATCH_QUEUE_CONCURRENT);
/// 代码段1
for (int i = 0; i<10000; i++) {
dispatch_async(queue, ^{
self.name = @"1234567890ABCDEFG"; // // 字面量创建的对象,值存在常量区,不会产生过度释放问题
NSLog(@"self.name %@",self.name);
});
}
/// 代码段2
for (int i = 0; i<10000; i++) {
dispatch_async(queue, ^{
self.name = [NSString stringWithFormat:@"CJL"]; // alloc 堆 iOS优化 - taggedpointer
NSLog(@"self.name %@",self.name);
});
}
/// 代码段3
for (int i = 0; i<10000; i++) {
dispatch_async(queue, ^{
self.name = [NSString stringWithFormat:@"1234567890ABCDEFG"];
NSLog(@"self.name %@",self.name);
});
}
}
1)先说结论吧
- 代码段1:
不会有任何问题, 无论字符串长度是多少,都不会有问题,此时name是__NSCFConstantString,字符串存放在常量区
- 代码段2/代码段3: 但由
数字、英文字母组合且长度超过9位
或者有中文或者其他字符
时,就会崩溃, 原理: 当符合上述条件时, 此时name是NSTaggedPointerString小对象
, 字符串值直接用地址表示,当不符合上述条件是,这个时候就是一个堆对象了__NSCFString
,此时多线程先问,就会多线程执行[_name release]
, 导致过度释放
,然后崩溃了
2)NSString的内存管理主要分为3种
__NSCFConstantString
:字符串常量,是一种编译时常量
,retainCount值很大,对其操作,不会引起引用计数变化,存储在字符串常量区
, 字面量创建的都是字符串常量, 推荐使用__NSCFString
:是在运行时
创建的NSString子类
,创建后引用计数会加1,存储在堆上
-
NSTaggedPointerString
:标签指针,是苹果在64位环境下对NSString、NSNumber
等对象做的优化。对于NSString对象来说当字符串是由
数字、英文字母
组合且长度小于等于9时,会自动成为
NSTaggedPointerString类型,存储在
常量区`当有
中文或者其他特殊符号
时,会直接成为__NSCFString
类型,存储在堆区
,但用字面量创建时,依然存储在常量区,为NSCFConstantString类型。
3). 获知NSString是否是NSTaggedPointerString,最简单的验证方式是: p/x 打印
- 直接看打印台:
- 看地址:
Tagged Pointer 标记:x86最后一位是标记位,arm64最高位是标记位。1表示是Tagged Pointer对象,0表示是普通对象。
4) 那为什么NSTaggedPointerString小对象不会多线程不会产生过度释放的问题呢?
我们来分析源码:
- 进入objc_retain、objc_release源码,在这里都判断是否是小对象,如果是小对象,则不会进行retain或者release,会直接返回。因此可以得出一个结论:如果对象是小对象,不会进行retain 和 release
//****************objc_retain****************
__attribute__((aligned(16), flatten, noinline))
id
objc_retain(id obj)
{
if (!obj) return obj;
//判断是否是小对象,如果是,则直接返回对象
if (obj->isTaggedPointer()) return obj;
//如果不是小对象,则retain
return obj->retain();
}
//****************objc_release****************
__attribute__((aligned(16), flatten, noinline))
void
objc_release(id obj)
{
if (!obj) return;
//如果是小对象,则直接返回
if (obj->isTaggedPointer()) return;
//如果不是小对象,则release
return obj->release();
}
小对象的地址分析
继续以NSString为例,对于NSString来说
一般的
NSString
对象指针,都是string值 + 指针地址
,两者是分开的对于
Tagged Pointer
指针,其指针+值
,都能在小对象中体现。所以Tagged Pointer
既包含指针,也包含值
在之前的文章讲类的加载时,其中的_read_images
源码有一个方法对小对象进行了处理,即initializeTaggedPointerObfuscator
方法
- 进入
_read_images -> initializeTaggedPointerObfuscator
源码实现
static void
initializeTaggedPointerObfuscator(void)
{
if (sdkIsOlderThan(10_14, 12_0, 12_0, 5_0, 3_0) ||
// Set the obfuscator to zero for apps linked against older SDKs,
// in case they're relying on the tagged pointer representation.
DisableTaggedPointerObfuscation) {
objc_debug_taggedpointer_obfuscator = 0;
}
//在iOS14之后,对小对象进行了混淆,通过与操作+_OBJC_TAG_MASK混淆
else {
// Pull random data into the variable, then shift away all non-payload bits.
arc4random_buf(&objc_debug_taggedpointer_obfuscator,
sizeof(objc_debug_taggedpointer_obfuscator));
objc_debug_taggedpointer_obfuscator &= ~_OBJC_TAG_MASK;
}
}
在实现中,我们可以看出,在iOS14之后,Tagged Pointer
采用了混淆处理, 无法直接从地址获得字符串值,如下所示
- 我们可以在源码中通过
objc_debug_taggedpointer_obfuscator
查找taggedPointer的编码
和解码
,来查看底层是如何混淆处理的
//编码
static inline void * _Nonnull
_objc_encodeTaggedPointer(uintptr_t ptr)
{
return (void *)(objc_debug_taggedpointer_obfuscator ^ ptr);
}
//编码
static inline uintptr_t
_objc_decodeTaggedPointer(const void * _Nullable ptr)
{
return (uintptr_t)ptr ^ objc_debug_taggedpointer_obfuscator;
通过实现,我们可以得知,在编码和解码部分,经过了两层异或
,其目的是得到小对象自己
,例如以 1010 0001
为例,假设mask
为 0101 1000
1010 0001
^0101 1000 mask(编码)
1111 1001
^0101 1000 mask(解码)
1010 0001
- 所以在外界,为了
获取小对象的真实地址
,我们可以将解码的源码拷贝到外面,将NSString混淆部分进行解码
,如下所示
观察解码后的小对象地址
,其中的62
表示b
的ASCII
码,
到这里,我们验证了小对象指针地址中确实存储了值
,那么小对象地址高位其中的0xa、0xb
又是什么含义呢?
//NSString
0xa000000000000621
//NSNumber
0xb000000000000012
0xb000000000000025
- 需要去源码中查看
_objc_isTaggedPointer
源码,主要是通过保留最高位的值
(即64位的值),判断是否等于_OBJC_TAG_MASK
(即2^63),来判断是否是小对象
static inline bool
_objc_isTaggedPointer(const void * _Nullable ptr)
{
//等价于 ptr & 1左移63,即2^63,相当于除了64位,其他位都为0,即只是保留了最高位的值
return ((uintptr_t)ptr & _OBJC_TAG_MASK) == _OBJC_TAG_MASK;
}
所以0xa、0xb
主要是用于判断是否是小对象taggedpointer,即判断条件
,判断第64位上是否为1(taggedpointer
指针地址即表示指针地址,也表示值)
0xa
转换成二进制为1 010
(64位为1,表示为taggedpointer,63~61后三位表示 tagType类型: 2,表示NSString
类型)0xb
转换为二进制为1 011
(64位为1,表示为taggedpointer,63~61后三位表示 tagType类型 : 3,表示NSNumber
类型),这里需要注意一点,如果NSNumber
的值是-1
,其地址中的值是用补码
表示的
这里可以通过_objc_makeTaggedPointer
方法的参数tag类型objc_tag_index_t
进入其枚举,其中 2
表示NSString
,3
表示NSNumber
-
同理,我们可以定义一个
NSDate对象
,来验证其tagType
是否为6
。通过打印结果,其地址高位是0xe
,转换为二进制为1 110
,排除64位的1,剩余的3位正好转换为十进制是6,符合上面的枚举值
Tagged Pointer 总结
Tagged Pointer
小对象类型(用于存储NSNumber、NSDate、小NSString
),小对象指针不再是简单的地址,而是地址 + 值
,即真正的值
,所以,实际上它不再是一个对象了,它只是一个披着对象皮的普通变量
而以。所以可以直接进行读取。优点是占用空间小 节省内存
Tagged Pointer
小对象不会进入retain 和 release
,而是直接返回了
,意味着不需要ARC进行管理
,所以可以直接被系统自主的释放和回收
Tagged Pointer
的内存并不存储在堆
中,而是在常量区
中,也不需要malloc和free
,所以可以直接读取,相比存储在堆区的数据读取,效率上快了3倍左右
。创建
的效率相比堆区快了近100倍左右
所以,综合来说,
taggedPointer
的内存管理方案,比常规的内存管理,要快很多Tagged Pointer
的64位地址中,前4
位代表类型
,后4位主要适用于系统做一些处理,中间56位用于存储值
优化内存建议:对于
NSString
来说,当字符串较小
时,建议直接通过@""
初始化,因为存储在常量区
,可以直接进行读取。会比WithFormat初始化方式
更加快速