11. 分类、扩展、协议、ARC

增强一个类的能力

在面向对象编程中,有两种方式可以增强一个类的功能,一种是继承,另一种是遵守一个协议。

继承:子类可以拥有父类所有的属性和

               方法,子类可以重写父类的方法。

                        

                老盖茨说这种勤奋与生俱来,从祖父

            辈到他,再到比尔.盖茨,最大的共同点

              就是勤奋。


缺点:由于继续关系使两个类的关系很紧密,不但继承了 

              父类的优点,也继承了父类的缺点。


遵守协议:遵守一个协议也可以加强

                      一个类的能力。


 缺点:类的本质发生改变。


有没有一种方式,既可增强一个类的能力,又可避免两个类的关系过于紧密、又不改变类的本质的方式呢?

一、分类

什么是分类Category

分类就是类的补充和扩展部分

补充和扩展的每个部分就是分类

分类本质上是类的一部分


分类的定义

分类也是以代码的形式保存在文件中

分类文件命名 主类类名+分类类名

分类文件也分为*.h文件和*.m文件

*.h文件存放分类的声明部分内容

@interface 主类类名(分类类名)

//添加方法声明

@end

.m文件存放分类的实现部分内容

@implementation 主类类名(分类类名)

//添加方法实现

@end

分类中是不可以创建实例变量的,自然也不可以创建自属性。

在分类中是可以访问主类的属性,但不可以访问主类的实例变量。


二、扩展(延展)

1.概念

扩展其实就是分类的一种特殊形式,扩展是没有名字的。


2.使用方式

a.扩展中可以声明实例变量,所以可以声明属性

b.扩展通常定义在文件的.m中,不能分开。

c.扩展是用来声明私有的属性和方法


区别:

分类:是不可以声明实例变量,通常是公开的,文件名通常为:"主类类名+分类类名.h"

扩展:是可以声明实例变量,是私有的,文件名通常为:"主类类名_扩展标识.h",注意扩展没有名的。


区别分类与扩展

1.都可以在主类中声明使用

2.通常来讲由于分类不能创建实例变化,本质上与主类有区别,所以不建议写在主类中。

3.扩展与主类紧密联系在一起,可以创建实例变量,所以通常来讲会把扩展和主类创建在一起。


三、协议

1.概念

协议就是规则,定义一个协议就相当于制定规则。

OC中类可以遵守协议,遵守了一个协议的类相当于拥有了一种能力。


2.语法

@protocal 协议名

@required 声明必须遵守的属性和方法

@optional 声明可选(可以)遵守的属性和方法

默认 @required

@end


3.一个类遵守一个协议

a.@interface 类名(分类类名):父类名<协议名>

b.实现协议中声明的方法

4.使用协议类型的引用指向实现了协议或者遵守了协议的对象

id<TRProtocol> p = [[MyClass]init];

[p …];可以向协议的引用发送消息,只能发送协议要求的消息。

5.协议的继承

协议的继承相当于协议的合并。

@protocol TRTarena2 <TRTarena>

-(void)learn;

   @end

6.一个类可以同时遵守多个协议,协议之间使用","分隔符分开。

     @interface TRStudent : NSObject<TRTarena,TRTarena3>

7.协议的使用和多态相类似,可以用于数组、参数、返回值类型,只不过多态返回的对象,一定要有继承关系,协议类型返回的对象,一定要有遵守协议或实现协议。


//
//  TRButtonProtocol.h
//  day06-4
//
//  Created by tarena on 14-3-24.
//  Copyright (c) 2014年 bamboo. All rights reserved.
//

#import <Foundation/Foundation.h>

@protocol TRButtonProtocol <NSObject>
-(void)onClick;
@end

//
//  TRPerson.h
//  day06-2
//
//  Created by tarena on 14-3-24.
//  Copyright (c) 2014年 bamboo. All rights reserved.
//

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

@interface TRPerson : NSObject <TRSpider>
-(void)job;
@end

//
//  TRPerson.m
//  day06-2
//
//  Created by tarena on 14-3-24.
//  Copyright (c) 2014年 bamboo. All rights reserved.
//

#import "TRPerson.h"
#import "TRPerson_TRExtention.h" //扩展 私有的
#import "TRSpider.h"

@implementation TRPerson
-(void)job{
    NSLog(@"人具有工作的能力");
}
-(void)addMethod{
    NSLog(@"添加了一个扩展的方法");
}
-(void)addMethod2{
    NSLog(@"添加了一个方法,协议。");
}
@end

//
//  TRButton2.h
//  day06-4
//
//  Created by tarena on 14-3-24.
//  Copyright (c) 2014年 bamboo. All rights reserved.
//

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

@interface TRButton2 : NSObject <TRButtonProtocol>

@end

//
//  TRButton2.m
//  day06-4
//
//  Created by tarena on 14-3-24.
//  Copyright (c) 2014年 bamboo. All rights reserved.
//

#import "TRButton2.h"

@implementation TRButton2
-(void)onClick{
    NSLog(@"注册");
}
@end

//
//  main.m
//  day06-4
//
//  Created by tarena on 14-3-24.
//  Copyright (c) 2014年 bamboo. All rights reserved.
//

#import <Foundation/Foundation.h>
#import "TRButtonProtocol.h"
#import "TRButton.h"
#import "TRButton2.h"
int main(int argc, const char * argv[])
{

    @autoreleasepool {
        
        //协议的使用和多态类似
        id<TRButtonProtocol> button = [[TRButton alloc]init];
        [button onClick];
        id<TRButtonProtocol> button2 = [[TRButton2 alloc]init];
        [button2 onClick];
        
    }
    return 0;
}

四、内存管理ARC自动引用计数器管理

1.ARC IOS5.0以后才支持,IOS7.0以后,强制使用ARC。

2.ARC "Automatic" Refercences Counting

3.原理

       依然使用引用计数器来管理内存,只是引用计数器的操作方式不同,由程序员发送消息转换为“编译器”帮我们自动发送消息,会在合适的位置自动加入retain、release、autorelease消息来进行计数管理,ARC是一种编译期语法。

4.使用ARC

a.在ARC中,程序中不能出现retain、release、autorelease…

b.在ARC中,程序不能在dealloc方法中显示调用父类的dealloc方法,一切在MRC中和内存相关的操作,ARC中都不能使用。


5.强引用

a.在程序中定义的引用,默认为强引用,所谓的强引用指向一个对象时,对象的引用计数器会自动加1,当引用超出作用域"{}",对象的引用计数器就会减1.

b.定义强引用

  __strong TRStudent* student = [TRStudent alloc]init];

c.当一个对象被引用指向时,此对象会隐式的retain一次,当强引用超出作用域时,指向的对象会隐式的发送release消息一次。

d.引用在使用的时候,会根据作用域的范围,自动做加1或减1的操作。

6.弱引用

a.定义弱引用

__weak TRStudent* student;

b.仅仅就是指向对象,但不会隐式的发送retain消息,出了作用域也不会发送realse消息。

c.XXX当一个弱引用指向的对象,未销毁时,向对象发送消息,此弱引用会自动的变为强引用。

d.当一个弱引用指向的对象被销毁时,弱引用本身会自动的赋值为空。(nil)

  zeroing weak reference

7.定义属性的时候,内存管理的描述

@property(nonatomic,strong)…;

@property(nonatomic,weak)…;


8.其它修饰关键字

@property(nonatomic,unsafe_unretained)int age;

a. unsafe_unretained等同于"assgin",功能和__weak几乎一样,唯一不同,没有"zeroing weak reference",通常使用在基本数据类型。

b.__autoreleaseing 用在方法的返回值,将返回值的对象放入到自动释放池中。

+(id)student{

__autoreleaseing TRStudent* student = [[TRStudent alloc]init];

    return student;

}


9.dealloc方法

      a.在ARC下,dealloc方法不允许调用父类的dealloc方法,当然也不允许向任何对象发送release消息,所以说dealloc方法几乎无用。

       b.在一些特殊情况下需要重写dealloc方法。

  1)在类中使用了C语言中的函数malloc分配内存。

  2)在类中使用了C++语言中的函数new等方式创建内存空间。

  此时需要在dealloc中对这些特殊的空间进行释放。

10.声明引用自动置空

a.在ARC下,如果定了一个引用没有赋值,编译会自动的初始化设置引用为空值。TRStudent* student;

b.为了尊重C语言的规范,基本数据类型没有初始化值,依然是垃圾值。

11.MRC和ARC混用

a.把MRC的代码转换成ARC的代码(手动)

    retain=>strong

    release、autorelease、[super dealloc]删除掉。

b.xcode提供了自动将MRC转换成ARC的功能。

  菜单栏(Edit)->Refacotor(重构)->Convert to Objective-C ARC

c.在ARC项目中继续使用MRC编译的类

    在编译选项中标识MRC文件即可

    "-fno-objc-arc"

d.在MRC项目中继续使用ARC编译的类

    在编译选项中标识MRC文件即可

    "-fobjc-arc"



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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值