文章标题
2015-09-24 17:33
302 查看
今天遇到了几个和字符串相关的内存问题,和大家分享一下
NSString *name = [[NSString alloc]initWithString:@”张三”];
NSLog(@”%d”,[name retainCount]);
这两行代码的打印结果是-1,
NSString * aString = [[NSString alloc] initWithFormat:@”123”];
NSLog(@”%d”,aString.retainCount);
这两行代码的打印结果也是-1
NSString * aString = [[NSString alloc] initWithFormat:@”1233sfsf4545f”];
NSLog(@”%d”,aString.retainCount);
当就字符串变为上面所述时,打印结果变为了1
这就奇怪了,为什么相同的语法打印的结果会是不同呢?反过来分析,打印结果是-1说明引用指向的是常量区的字符串,打印结果是1指向的是堆区的字符串。由于OC是不开源的,内部的实现我们不得而知,但是从中我们至少可以猜测,OC中对不同大小的字符串是由不同的方法的。
initWithString产生的是将指针指向了常量区的字符串,是无法被release的,如果使用dealloc进行摧毁会报错。其一:不能手动调用dealloc方法 再者苹果官方文档中说的很清楚,创建的对象和retain的对象为自己所保有,这些对象全部都是在堆区的。静态区的内存实在编译时就分配好了的,它的内存地址非常靠前,而且在程序运行的整个阶段都存在,所以我们不能释放。
关于类簇(class cluster)大家举得最多的例子就是NSNumber类,其实NSNumber类是一个抽象的超类,内部有很多的具体的子类,如NSInt NSDouble等,它们对应不同的初始化方法,也就是说NSNumber的不同初始化方法返回的类型是不同的。不仅NSNumber,NSString也是如此,
// 类簇的使用
id someClass = [NSString alloc]; // 返回的对象类型:NSPlaceholderString
NSString *string1 = [[NSString alloc] init]; // 返回的对象类型:__NSCFConstantString
NSString *string2 = [[NSString alloc] initWithFormat:@”string2”]; // 返回的对象类型:__NSCFString
NSLog(@”%@”, string1);
NSLog(@”%@”, string2);
类簇可以简化一个面向对象的公开架构,而又不减少功能的丰富性
我们在项目中肯定会遇到类的初始化方法传参的情况,如果实在MRC模式下,如何保证内存不leak,苹果的官方文档做了如下推荐
(id)initWithName:(NSString*)name{
self = [super init];
if(self){
_name = [name copy];// 当然name属性的语义控制要使用copy
}
return self;
}
使用self.name = name;其实和上面是相同的
NSString *name = [[NSString alloc]initWithString:@”张三”];
NSLog(@”%d”,[name retainCount]);
这两行代码的打印结果是-1,
NSString * aString = [[NSString alloc] initWithFormat:@”123”];
NSLog(@”%d”,aString.retainCount);
这两行代码的打印结果也是-1
NSString * aString = [[NSString alloc] initWithFormat:@”1233sfsf4545f”];
NSLog(@”%d”,aString.retainCount);
当就字符串变为上面所述时,打印结果变为了1
这就奇怪了,为什么相同的语法打印的结果会是不同呢?反过来分析,打印结果是-1说明引用指向的是常量区的字符串,打印结果是1指向的是堆区的字符串。由于OC是不开源的,内部的实现我们不得而知,但是从中我们至少可以猜测,OC中对不同大小的字符串是由不同的方法的。
initWithString产生的是将指针指向了常量区的字符串,是无法被release的,如果使用dealloc进行摧毁会报错。其一:不能手动调用dealloc方法 再者苹果官方文档中说的很清楚,创建的对象和retain的对象为自己所保有,这些对象全部都是在堆区的。静态区的内存实在编译时就分配好了的,它的内存地址非常靠前,而且在程序运行的整个阶段都存在,所以我们不能释放。
关于类簇(class cluster)大家举得最多的例子就是NSNumber类,其实NSNumber类是一个抽象的超类,内部有很多的具体的子类,如NSInt NSDouble等,它们对应不同的初始化方法,也就是说NSNumber的不同初始化方法返回的类型是不同的。不仅NSNumber,NSString也是如此,
// 类簇的使用
id someClass = [NSString alloc]; // 返回的对象类型:NSPlaceholderString
NSString *string1 = [[NSString alloc] init]; // 返回的对象类型:__NSCFConstantString
NSString *string2 = [[NSString alloc] initWithFormat:@”string2”]; // 返回的对象类型:__NSCFString
NSLog(@”%@”, string1);
NSLog(@”%@”, string2);
类簇可以简化一个面向对象的公开架构,而又不减少功能的丰富性
我们在项目中肯定会遇到类的初始化方法传参的情况,如果实在MRC模式下,如何保证内存不leak,苹果的官方文档做了如下推荐
(id)initWithName:(NSString*)name{
self = [super init];
if(self){
_name = [name copy];// 当然name属性的语义控制要使用copy
}
return self;
}
使用self.name = name;其实和上面是相同的
相关文章推荐
- ROS 学习系列 -- RViz中移动机器人来学习 URDF,TF,base_link, map,odom和odom 主题的关系
- 控件
- IDEA 导出自己的jar包 并且在另一个工程中引用
- html里的常用特殊符号表示大全
- 第一篇博客
- multithreading
- Ubuntu14.04安装JDK与配置环境变量
- 讨论一下用友ERP软件系统的优缺点与集成业务软件系统的优势
- 初步学习lex和yacc
- POJ 1273 Drainage Ditches(最大流模板)
- Timer与TimerTask相关
- [9-15]Sed文本处理――高级用法
- pushMeBaby,github链接
- 整理的页面切换效果,项目源码。请大家收藏!
- 图像处理顶级期刊
- Android ant自动打包 crunch 报错
- ☆第二次作业
- 表单提交验证
- Guava Cache 实现详解
- Autolayout 基础