OC中的类簇的使用和初始化方法中属性的使用-创新互联

今天遇到了几个和字符串相关的内存问题,和大家分享一下

创新互联公司长期为上1000家客户提供的网站建设服务,团队从业经验10年,关注不同地域、不同群体,并针对不同对象提供差异化的产品和服务;打造开放共赢平台,与合作伙伴共同营造健康的互联网生态环境。为椒江企业提供专业的做网站、网站设计椒江网站改版等技术服务。拥有十多年丰富建站经验和众多成功案例,为您定制开发。

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;其实和上面是相同的

创新互联www.cdcxhl.cn,专业提供香港、美国云服务器,动态BGP最优骨干路由自动选择,持续稳定高效的网络助力业务部署。公司持有工信部办法的idc、isp许可证, 机房独有T级流量清洗系统配攻击溯源,准确进行流量调度,确保服务器高可用性。佳节活动现已开启,新人活动云服务器买多久送多久。


网站名称:OC中的类簇的使用和初始化方法中属性的使用-创新互联
浏览地址:http://csdahua.cn/article/cedeje.html
扫二维码与项目经理沟通

我们在微信上24小时期待你的声音

解答本文疑问/技术咨询/运营咨询/技术建议/互联网交流