关于autoreleasepool一个常见的面试题

面试题如下:

for (int i = 0; i < 10; ++i) {
   NSString *str = @"Hello World";
   str = [str stringByAppendingFormat:@" - %d", i];
   str = [str uppercaseString];
   NSLog(@"%@", str);
}
问:以上代码存在什么样的问题?如果循环的次数变大时,应该如何改进?


有些人可能认为错误在第3行,因为是NSString(不可变字符串),在第3行对字符串重新赋值,所以他认为这里有错。

可以肯定第3行没有错误,没有搞明白的童鞋请加强OC基础的学习。


在ios开发中,并没有java或者C#中的垃圾回收机制,

虽然我们使用ARC开发,但是ARC只是在编译时,编译器会根据代码结构自动添加retain、release和autorelease

自动释放池(autoreleasepool)的工作原理:

标记为autorelease的对象在出了作用域范围后,会被添加到最近一次创建的自动释放池中

当自动释放池被销毁或耗尽时,会向自动释放池中的所有对象发送release消息

也就是说对象在出了作用域之后不一定马上被销毁


我们再来看上面的代码,如果循环次数变大,而NSString对象并没有马上被释放,在内存中会同时存在很多个NSString对象,造成内存溢出

1,如果循环次数比较大,可改进为:

@autoreleasepool {
        
    for (int i = 0; i < 50; i++) {
        NSString *str = @"Hello World";
        
        str = [str uppercaseString];
        
        str = [NSString stringWithFormat:@"%@ %d", str, i];
        
        NSLog(@"%@", str);
    }
}
这样在整个for循环执行完毕之后,for循环中创建的所有NSString对象都会被销毁。

2,如果循环次数非常大,可改进为:

for (int i = 0; i < 1000; i++) {
    @autoreleasepool {
        NSString *str = @"Hello World";
        
        str = [str uppercaseString];
        
        str = [NSString stringWithFormat:@"%@ %d", str, i];
        
        NSLog(@"%@", str);
    }
}
这样改进,单次循环执行完毕后,创建的NSString对象会及时销毁。


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值