- (void)viewDidLoad {
[super viewDidLoad];
self.view.backgroundColor = [UIColor lightGrayColor];
[self test1];
}
- (void)test {
//问:以上代码存在什么样的问题?如果循环的次数变大时,应该如何改进?
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,如果循环次数比较大,可改进为:
- (void)test1 {
@autoreleasepool {
for (int i = 0; i < 1000000; i++) {
NSString *str = @"Hello World";
str = [str uppercaseString];
str = [NSString stringWithFormat:@"%@ %d", str, i];
NSLog(@"%@", str);
}
}
}
//这样在整个for循环执行完毕之后,for循环中创建的所有NSString对象都会被销毁。
//2,如果循环次数非常大,可改进为:
- (void)test2 {
for (int i = 0; i < 1000000; i++) {
@autoreleasepool {
NSString *str = @"Hello World";
str = [str uppercaseString];
str = [NSString stringWithFormat:@"%@ %d", str, i];
NSLog(@"%@", str);
}
}
}
//这样改进,单次循环执行完毕后,创建的NSString对象会及时销毁。
网友评论