OC中 NSString引用计数器和initWithFormat:(NSString *)string中String的存储区

前几天OC试题测验,关于NSString  的计数机制考试了好几道题,又在自己实验的时候发现格式化初始字符串时,字符串存储在哪里是和字符串的复杂度有关的因此我觉得有必要整理一下。

引子:

在Objective-C中的内存管理是通过一种叫做“引用计数器”的机制管理的。

举例, 当声明了一个新的实例

NSData *data = [[NSData alloc] init];

现在,内存中有一个NSData类型的对象,名叫data。它的引用计数器的值为1. 如果我再次引用它的时候,他的引用计数值会+1变为2.

使用完毕后需要调用

[data release];

来使引用计数-1.  当该值为零的时候,系统会将data实例回收掉并释放内存。

但当创建的对象是

代码如下:

NSString *str = [[NSString alloc] initWithString:@"ABC"];

str = @"123";

[str release];

NSLog(@"%@",str);
 
首先,对这段代码进行分析。
第一句 声明了一个NSString类型的实例 str,初始化后init后,其内部的字符串"ABC"在堆区。格式化初始字符串我试验了下发现"A-Z,a-z"retainCount都还是 -1,而“AA,aa”一旦字符超过一个就会在堆区创建,retainCount 都会变成 1.
第二行,将str的指针指向了一个常量@"123"。 理论上讲在第一行初始化的@"ABC"没有任何任何指针指向了。 所以造成了内存泄露
然后第三行, 将str的引用计数-1 
第四行输出str的值  为123.

为什么不会崩溃? 因为第三行的release 实际上是release了一个常量@"123"  而作为常量,其默认的引用计数值是整形最大值。因为超出了短整形的范围。所以采用%d ,retainCount输出时,输出值为 -1.
NSLog(@"retainCount = %d",[@"123" retainCount]);
最终的输出值是 -1若用%ld 输出,是整形最大值。  所以单单一个release是不会将其释放掉的。
然后再回答这样会不会造成内存泄露。
理论上 会。  
但是实际上,Objective-C对NSString类型有特殊照顾。所有的NSString的引用计数器默认初始值都会非常非常大。

在学习C的时候我们知道常量中有,字符串常量这一种,基本数据类型也是常量。它们都存放在常量区,一旦创建后其值不可改变,比如你不能这样写:3 = 5。 这样是错误的。因此为了不让OC字符串改变引发问题,我猜想,这才是,给存放在常量区的字符串计数器默认值初始值的原因。因为这样默认后,不管你怎么release ,都是没有办法释放掉常量的这块内存的。

 

生成一个NSString类型的字符串有三种方法:

方法1.直接赋值:     NSString *str1 = @"my string"

 

方法2.类方法 初始化生成:     NSString *str2 = [NSString stringWithString:@"my string"];

 

方法3.实例方法 初始化生成:  非格式化:  NSString *str3 = [[NSString alloc] initWithString:@"my string"];

               格式化: NSString *str4 = [[NSString alloc]initWithFormat:@"my string"];

 

区别1: 方法一生成字符串时,不会初始化内存空间,所以使用结束后不用释放内存;

   而其他三个都会初始化内存空间,使用结束后要释放内存;

   在释放内存时方法2和3也不同,方法2是autorelease类型,内存由系统释放;方法3则必须手动释放

区别2:用Format初始化的字符串,需要初始化一段动态内存空间,如:0x6a42a40;

   而用String声明的字符串,初始化的是常量内存区,如:0x46a8常量内存区的地址,只要值相同,占用的地址空间是一致的。

   所以str3和str1的地址一致,但是str4和str1的地址不一致。

大家可以自行实验验证一下。

如果例子中的不是NSString而是其他类型 例如NSData、NSNumber等等等。。。

则肯定会造成内存泄露。

内存泄露的后果在短时间内不会显现。 但以长远角度来看,后果是非常非常严重的。
同学们一定要避免内存泄露的发生。
若想这样,那就必须要养成良好的编码习惯。
引用官方文档的一句话说,
谁引用了谁负责释放
说白了就是谁Init、Retain、Copy了。 谁负责Release。
理论上讲,你代码中的init+retain+copy的个数 应该等于release的个数。
这样才可以最大程度上的避免内存泄露的发生。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值