码迷,mamicode.com
首页 > 其他好文 > 详细

OC内存管理

时间:2015-08-13 21:43:11      阅读:102      评论:0      收藏:0      [点我收藏+]

标签:

新来报道!欢迎指导纠错!
引言:
     1、OC中的对象都是分配在堆中的
               声明对象的格式:
                    Person *person = [Person new];
                    Person *person           //指针类型的变量是放在栈里面的;
                    [Person new]               //在堆创建对象并初始化;                    
                    /*用指针变量指向了堆中创建出来的对象*/
                    Person *person2 = person;
                    /*person2/person都指向了同一个对象*/
     2、堆里的对象应该谁创建、谁释放
          ARC     自动引用计数
            MRC    手动引用计数
     3、什么是引用计数?
           在程序当中出现多个指针指向同一个对象的情况。在这种情况下要保障不能提前释放这个对象,必须是所有指针都不再使用这个对象的时候才能释放对象。
          对象中存储被引用的次数,当被引用时候计数器加1,不再引用时计数器减1,当计数器为0的时候,真正销毁对象。
     4、对象所有权
          指针在指向一个对象时,可以获得对象所有权(引用计数会加1);也可以不获得对象所有权(引用计数不发生变化)。
          ARC:strong、weak、assign。
          MRC:retain、assign。 
 
dealloc 是用来销毁对象所占的内存,而release是用来使对象的引用计数减一
 
总结:
内存管理、实例变量
1、为何引用内存管理?
内存管理是程序设计中常见的资源管理的一部分。每个计算机系统可供程序使用的资源都是有限的,这些资源包括内存、打开文件数量以及网络连接等。如果你使用了某种资源,例如因打开文件而占用了资源,那么你需要随后进行清理。如果你不断打开文件并且保持打开状态而且不去关闭,最终将无法打开新的文件。摄像一下公共图书馆的场景。任何人都只借不还,最终图书馆将会因无书而倒闭,每个人都无法再使用图书馆。
1)对象生命周期
对象的生命周期包括诞生(通过alloc或new方法实现)、生存(接收消息和执行操作)、交友(借助方法的组合和参数)以及当他们的生命结束时最终死去(释放)。当对象的生命周期结束时,他们的源材料(内存)将被回收以供新的对象使用。
2)引用计数
Cocoa采用了一种称为引用计数的技术,有时也叫做保留计数。每个对象有一个与之相关联的整数,称作为引用计数器或保留计数器。当某段代码需要访问一个对象时,该代码将该对象的保留计数器加1,表示“我要访问该对象”。当这段代码结束对象访问时,将对象的保留计数器值减1,表示不再访问该对象。当保留计数器值为0时,表示不再有代码访问该对象了,因此对象将被销毁,其占用的内存被系统回收以便重用。
当使用alloc、new方法或者通过copy消息(生成接收对象的一个副本)创建一个对象时,对象的保留计数器值被设置为1.
以下:retainCount会加1;
1.Person *person =[ [Person alloc]init];
2.Person *person = [Person new];
3.NSMutableString *array = [NSMuatbleString stringWithString :@“”];
  [array setObjetct:person];
4、[person retain];
注意:Person *person2 = person; //不会获得对象所有权
以下:retainCount会减1:
1.[person release];
2.[array removeObject:person];
 
2、Cocoa内存管理规则
1.如果使用new、alloc或copy操作获得一个对象,则该对象的保留计数器值为1.
2.如果通过任何其他方法获得一个对象,则假设该对象的保留计数器值为1,而且已经被设置为自动释放。
3.如果保留了某个对象,则必须保持retain方法和release方法的的使用次数相等。
 
 
代码注意:
1.self指向的是实例的一个指针,不能在类方法中使用
+(Person *)personWithName:(NSString *)name andAge:(NSInteger)age
{
self = [super init];
if(self)
{
_name = name;
_age = age;
}
return self;
}
2.类方法:
1)在类方法中,由于还没有创建对象实例,所以:self指针不能使用;一般类方法这么使用:

/*类方法:

     1.在类方法中,由于还没有创建对象实例,所以:self指针不能使用;

     2.实例变量不能在类方法中使用;*/

+(Person *)personWithName:(NSString *)name andAge:(NSInteger)age

{

     //类方法是通过类名使用的,没有创建对象实例变量

     return [[Person alloc]initWithName:name andAge:age];

     //类方法中的返回是具体的类型;

}

2)实例变量不能在类方法中使用;
3)dealloc在释放是自动调用,创建对象时是先创建基类部分,然后创建子类部分;销毁时先从子类销毁,然后销毁基类;ARC禁止显示发送deallo消息;
3、不可变对象不适用于引用计数
1.不可变对象是可变对象的基类
2.类中的对象型成员变量,他的set方法要保证获得对象所有权
-(void)setName:(NSString *)name
{
if(_name != name )
{
//要将原来对象的所有权释放掉
[_name release];
//在获得新的对象的所有权
_name = [name retain];//保留一个name的计数,获得所有权;
}
}
3)如果类中包含了对象类型的成员变量,必须重写dealloc方法,保证对象类型变量的正常释放;
-(void)dealloc
{
 
[super dealloc];
}
 
 附录:
//测试手动计数
        //1.创建对象会获得对象所有权
        Integer *i1 = [Integer integerWithInteger:10];
        NSLog(@"retainCount = %lu",[i1 retainCount]);
       
        //2.通过指针赋值,不获得对象所有权;
        Integer *i2 = i1;//不获得对象所有权
        NSLog(@"retainCount = %lu",[i2 retainCount]);
       
        //3.通过retain获得对象所有权
        [i1 retain];//获得对象所有权
        NSLog(@"retainCount = %lu",[i1 retainCount]);
       
        //4.将对象添加到容器中,获得对象所有权,容器中会获得对象的一个引用
        NSMutableArray *arr = [NSMutableArray array];
        [arr addObject:i1];
        NSLog(@"retainCount = %lu",[i1 retainCount]);
       
        //5.通过releas释放对象所有权,
        [i1 release];
        NSLog(@"retainCount = %lu",[i1 retainCount]);
       
        //6.从容器当中删除对象,也会释放对象所有权
        [arr removeObject:i1];
        NSLog(@"retainCount = %lu",[i1 retainCount]);
       
        //7.最后再释放一次,对象才会被正常销毁
        [i1 release];
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

OC内存管理

标签:

原文地址:http://www.cnblogs.com/xjf125/p/4728327.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!