美文网首页
01 Object-C之MRC

01 Object-C之MRC

作者: 紫苓 | 来源:发表于2018-01-25 13:46 被阅读15次

一直想着抽点时间把object-c学习一下,因为项目的原因断断续续的看了一些,但感觉都不是很系统,从网上扒了一个视频,趁着年底项目不是很紧张的时机,赶紧来跟着系统的学习一下。

1、MRC机制下的内存管理

MRC(Mannul Reference Counting):手动内存管理,在iPhone4(IOS4.3)之前也就是ARC(Automatic Reference Counting)出来之前,object -c中的对象都是需要手动进行回收的。

1)系统回收对象的判断依据是:对象下的retainCount是否为0,如果为0,则立刻进行回收,如果不为0,则不进行回收。
object-c为对象提供了两个方法来控制retainCount的值,[obj retain]可以使retainCount的值加1,[obj release]可以使retainCount的值减1,通过这两个方法,在MRC下可以实现对象的释放。

2)开发中的配对原则:只要出现alloc、retain、new等关键字,则应该有相应的release与之相匹配

3)@autoreleasepool和autorelease:在autoreleasepool作用域中,对象通过[obj autorelease]把对象本身加入到release pool中,在autoreleasepool作用域结束的时候,会对release pool中的所有对象统一调用一次release,从而实现对象的释放。autorelease一般会用在类方法中,或者是用来创建某个新对象的方法中。

4)@property注解参数
1、线程相关:
nonatomic:线程无关
atomic:线程有关,这个是默认值,但却不是IOS开发需要的
2、内存相关:
retain:生成的set/get方法是符合内存管理的,这个只能修饰对象数据类型
assign:生成的set/get方法是直接进行赋值的,可以修饰对象和基本数据类型
3、get/set方法相关:
readonly:只生成get方法,不生成set方法
readwrite:set/get方法都生成
setter="":重命名set方法,一般只会用在bool变量上面
getter="":重命名get方法,一般只会用在bool变量上面

5)开发中两个问题:野指针和内存泄漏
如果对某个对象进行了过多的release操作,带来的问题就是 EXC_BAD_ACCESS异常,也就是野指针操作;通常的做法是给release后的对象赋值为nil,因为在object-c的世界里没有nullpointexception这个概念。
如果对某个对象执行少了release操作,那就会导致对象不会被回收,从而带来内存泄漏问题。

2、retain/release 示例
//
//  Person.h
//  IOS_Study_Sample
//
//  Created by luozheng on 2018/1/25.
//  Copyright © 2018年 luozheng. All rights reserved.
//

#import <Foundation/Foundation.h>
#import "Car.h"

@interface Person : NSObject
{
    Car * _biaozhi408;
}

@property (nonatomic,retain,readwrite) Car * baoma5x;
@property (nonatomic,assign,readwrite) int age;

-(void) setBiaozhi408:(Car *) biaozhi408;
-(Car *) biaozhi408;

@end


//
//  Person.m
//  IOS_Study_Sample
//
//  Created by luozheng on 2018/1/25.
//  Copyright © 2018年 luozheng. All rights reserved.
//

#import "Person.h"

@implementation Person

/**
 * 记住这里的调用顺序,要先释放掉子类也就是Person类持有的资源,
 * 然后再调用[super dealloc]释放父类持有的资源。
 * 切记不要先调用[super dealloc]销毁父类资源,
 * 否则的话这里在释放子类资源的时候会抛EXC_BAD_ACCESS异常
 */
- (void)dealloc
{
    
    [_baoma5x release];
    
    [_biaozhi408 release];
    
    [super dealloc];
    NSLog(@"Person (age = %d)被回收了!!!!!",_age);
}

-(void) setBiaozhi408:(Car *) biaozhi408
{
    if(_biaozhi408 != biaozhi408)
    {
        _biaozhi408 = [biaozhi408 retain];
    }
}

-(Car *) biaozhi408
{
    return _biaozhi408;
}

@end


//
//  Car.h
//  IOS_Study_Sample
//
//  Created by luozheng on 2018/1/25.
//  Copyright © 2018年 luozheng. All rights reserved.
//

#import <Foundation/Foundation.h>

@interface Car : NSObject

@property (nonatomic,assign,readwrite) float cost;

@end


//
//  Car.m
//  IOS_Study_Sample
//
//  Created by luozheng on 2018/1/25.
//  Copyright © 2018年 luozheng. All rights reserved.
//

#import "Car.h"

@implementation Car

- (void)dealloc
{
    [super dealloc];
    NSLog(@"Car ( cost = %f ) 被回收了!",_cost);
}

@end


//
//  main.m
//  01-内存管理示例
//
//  Created by luozheng on 2018/1/25.
//  Copyright © 2018年 luozheng. All rights reserved.
//

#import <Foundation/Foundation.h>
#import "Person.h"

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        
        Person * person = [[Person alloc] init];
        person.age = 35;//这个真实是这样子的:[person setAge:35],不要认为是直接赋值
        
        Car * car1 = [[Car alloc] init];
        car1.cost = 50000.0;
        [person setBiaozhi408:car1];
        [car1 release];
        
        
        Car * car2 = [[Car alloc] init];
        car2.cost = 800000.0;
        [person setBaoma5x:car2];
        [car2 release];
        
        [person release];
        
        
    }
    return 0;
}

代码比较简单,主要要注意点就是在Person dealloc 方法中super dealloc 调用的位置上,要记住先销毁子类资源,再销毁父类资源。

3、@autoreleasepool/autorelease 示例

//
//  Person.h
//  IOS_Study_Sample
//
//  Created by luozheng on 2018/1/25.
//  Copyright © 2018年 luozheng. All rights reserved.
//

#import <Foundation/Foundation.h>

@interface Person : NSObject

@property(nonatomic,assign,readwrite) int age;

+(Person *) initWithAge:(int) age;

@end


//
//  Person.m
//  IOS_Study_Sample
//
//  Created by luozheng on 2018/1/25.
//  Copyright © 2018年 luozheng. All rights reserved.
//

#import "Person.h"

@implementation Person

- (void)dealloc
{
    [super dealloc];
    NSLog(@"Person (age = %d) 被销毁了 !!!! ",_age);
}

+(Person *) initWithAge:(int) age
{
    Person *person = [[self alloc] init];
    person.age = age;
    return [person autorelease];
}

@end


//
//  main.m
//  02-autorelease测试
//
//  Created by luozheng on 2018/1/25.
//  Copyright © 2018年 luozheng. All rights reserved.
//

#import <Foundation/Foundation.h>
#import "Person.h"

int main(int argc, const char * argv[]) {
    
    @autoreleasepool {
        // 这里为类方法,在类方法中创建了一个Person对象,并且这个对象在创建后
        // 通过autorelease加入到了release pool中。
        // aurelease方法一定要在@autoreleasepool作用域内调用,只有这样对象才会被放置到release pool中。
        Person *p1 = [Person initWithAge:36];
        NSLog(@"person age = %d",p1.age);
        
        Person *p2 = [Person initWithAge:66];
        NSLog(@"person age = %d",p2.age);
    }// 在@autoreleasepool作用域结束的时候,会对release pool中的所有对象统一调用一次release,
    // 这里就相当于:[p1 release],[p2 release];
    
    return 0;
}

4、特殊对象:NSString处理

NSString也是一个对象,在MRC机制下也需要进行手动管理,但是其稍微有些特殊。

/**
         * 字面量创建NSString对象,由系统进行管理
         */
        NSString *str1_0 = @"123";
        NSLog(@"str1_0 字面量赋值 retainCount = %li",(long)str1_0.retainCount);
        NSString *str1_1 = @"123456123456123456";
        NSLog(@"str1_1 字面量赋值 retainCount = %li",(long)str1_1.retainCount);
        NSString *str1_0_1 = @"中文";
        NSLog(@"str1_0_1 字面量赋值 中文 retainCount = %li",(long)str1_0_1.retainCount);
        NSString *str1_1_1 = @"赵昂问啊都快放假啊看风景的多。 到底 就。打算看就卡死多";
        NSLog(@"str1_1_1 字面量赋值 中文 retainCount = %li",(long)str1_1_1.retainCount);
        
        NSString *str1_2 = [[NSString alloc] initWithString:@"123"];
        NSLog(@"str1_2 initWithString retainCount = %li",(long)str1_2.retainCount);
        NSString *str1_3 = [[NSString alloc] initWithString:@"123456789123456789"];
        NSLog(@"str1_3 initWithString retainCount = %li",(long)str1_3.retainCount);
        NSString *str1_2_1 = [[NSString alloc] initWithString:@"中文"];
        NSLog(@"str1_2_1 initWithString 中文 retainCount = %li",(long)str1_2_1.retainCount);
        NSString *str1_3_1 = [[NSString alloc] initWithString:@"中文说的可激发思考的佛教萨看风景的肯"];
        NSLog(@"str1_3_1 initWithString 中文 retainCount = %li",(long)str1_3_1.retainCount);
        
        /**
         * stringWithFormat是类方法,基于autoreleasepool和autorelease机制,
         * 通过这种方式创建出来的对象是无需进行手动释放的。
         */
        NSString *str1_4 = [NSString stringWithFormat:@"%@",@"123"];
        NSLog(@"str1_4 stringWithFormat retainCount = %li",(long)str1_4.retainCount);
        NSString *str1_5 = [NSString stringWithFormat:@"%@",@"123456789123456789"];
        NSLog(@"str1_5 stringWithFormat retainCount = %li",(long)str1_5.retainCount);
        NSString *str1_4_1 = [NSString stringWithFormat:@"%@",@"你好啊"];
        NSLog(@"str1_4_1 stringWithFormat 中文 retainCount = %li",(long)str1_4_1.retainCount);
        NSString *str1_5_1 = [NSString stringWithFormat:@"%@",@"哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈"];
        NSLog(@"str1_5_1 stringWithFormat 中文 retainCount = %li",(long)str1_5_1.retainCount);
        
        /*
         * initWithFormat方式创建的对象,这种方式创建的对象要手动进行管理
         */
        NSString *str1_6 = [[NSString alloc] initWithFormat:@"%@",@"123"];
        NSLog(@"str1_6 initWithFormat retainCount = %li",(long)str1_6.retainCount);
        NSString *str1_7 = [[NSString alloc] initWithFormat:@"%@",@"123456789012"];
        NSLog(@"str1_7 initWithFormat retainCount = %li",(long)str1_7.retainCount);
        NSString *str1_6_1= [[NSString alloc] initWithFormat:@"%@",@"你好你好"];
        NSLog(@"str1_6_1 initWithFormat 中文 retainCount = %li",(long)str1_6_1.retainCount);
        NSString *str1_7_1 = [[NSString alloc] initWithFormat:@"%@",@"哈哈哈哈哈哈哈哈哈哈哈哈哈哈"];
        NSLog(@"str1_7_1 initWithFormat 中文 retainCount = %li",(long)str1_7_1.retainCount);
        
        /*
         * 综上,为了减少不必要的麻烦,对于NSString对象的创建,
         * 要么使用字面量创建,要么使用类方法stringWithFormat来创建,
         * 基于这两种创建方式创建的NSString对象,都不需要我们进行手动进行回收。
         */

测试结果:

str1_0 字面量赋值 retainCount = -1
str1_1 字面量赋值 retainCount = -1
str1_0_1 字面量赋值 中文 retainCount = -1
str1_1_1 字面量赋值 中文 retainCount = -1
str1_2 initWithString retainCount = -1
str1_3 initWithString retainCount = -1
str1_2_1 initWithString 中文 retainCount = -1
str1_3_1 initWithString 中文 retainCount = -1
str1_4 stringWithFormat retainCount = -1
str1_5 stringWithFormat retainCount = 1
str1_4_1 stringWithFormat 中文 retainCount = 1
str1_5_1 stringWithFormat 中文 retainCount = 1
str1_6 initWithFormat retainCount = -1
str1_7 initWithFormat retainCount = 1
str1_6_1 initWithFormat 中文 retainCount = 1
str1_7_1 initWithFormat 中文 retainCount = 1

根据测试结果,对于NSString对象的创建,要么采用字面量方式,要么采用类方法stringWithFormat方式,这样可以让系统来管理NSString,而不需要我们自己来进行手动管理。

相关文章

  • 01 Object-C之MRC

    一直想着抽点时间把object-c学习一下,因为项目的原因断断续续的看了一些,但感觉都不是很系统,从网上扒了一个视...

  • 静态内存分析-场景演练-桥接

    1.MRC环境下桥接 - (void)MRC{ //MRC下桥接 //Foundation到CoreFoundat...

  • @autoreleasepool 用法

    ****MRC**** 与 ****ARC********MRC****(****Mannul Reference...

  • 内存管理之MRC

    原本想写点别的,但是思来想去以内存管理做为开篇,一是现在多是用ARC的项目,怕时间长了也忘了这最根本的东西了;...

  • IOS之MRC ARC

    作者是以前搞Android的,用的是java语言,对象的释放都是由虚拟机完成,IOS用的是Object C对象需要...

  • Effective Object-C 52:1-5

    一、熟悉Object-C Object-C : 基于C语言基础 + 面向对象特性。 1.了解Object-C 的起...

  • MRC,ARC混编环境配置

    MRC --- ARC(让ARC文件在MRC的环境下进行ARC编译) -fobjc-arc ARC --- MRC...

  • iOS

    一、Object-c试题总结: 1、Object-c的类可以多继承么?可以实现多个接口么? 答:Object-c的...

  • .mm文件

    .m文件是纯Object-C 文件.mm是Object-C和C++混合文件 .m只能调用纯Object-C的类,不...

  • ARC MRC的转换

    ARC—>MRC 例:SDWebImage 回车输入 -fobjc-arc 回车 MRC—>ARC:

网友评论

      本文标题:01 Object-C之MRC

      本文链接:https://www.haomeiwen.com/subject/piuyaxtx.html