类别(Category)与扩展(Extensions)

时间:2021-07-24 17:07:43

一.类别(Category)

类别(Category)是一种可以为现有的类(包括类簇:NSString...,甚至源码无法获得的类)添加新方法的方式无需从现有的类继承子类。类别添加的新方法可以被子类继承。

注:继承(inheritance)无法为一个类簇创建子类。类别不能添加实例变量。

1. 创建类别

1.1 声明类别

类别的声明和类的声明格式相似:

@interface ClassName(CategoryName)

//method declarations

@end//CategoryName

       头文件"NSString+Tools.h":

  #import <Cocoa/Cocoa.h>

  @interface NSString (Tools)

  - (NSNumber *) lengthAsNumber;

  @end//Tools

1.2 实现类别

        实现文件"NSString+Tools.m":

  #import "NSString+Tools.h"

  @implementation NSString(Tools)

  - (NSNumber *) lengthAsNumber

  {

   unsigned int length = [self length];

return [NSNumber numberWithUnsignedInt: length];

  }

  @end//Tools

2. 类别的作用

类别有以下5个主要作用:

①为现有的类添加新方法;

②将类的实现分散到多个不同文件或多个不同框架中(把一个大的类按功能划分成几块,便于维护);

③创建对私有方法的前向应用;

④使用category的非正式协议(informal protocol)来实现委托(delegation);

⑤通过替换现有类中的方法,修正现有类(没有源码文件的情况下)的功能或错误。

2.1 把类按功能划分成几块

UIKIT_CLASS_AVAILABLE(2_0) @interface UIView : UIResponder<NSCoding, UIAppearance, UIAppearanceContainer> {

................................

}

.................................

@end//UIView

@interface UIView(UIViewGeometry)

.................

@end//UIViewGeometry

@interface UIView(UIViewHierarchy)

.................

@end//UIViewHierarchy

@interface UIView(UIViewRendering)

................

@end//UIViewRendering

@interface UIView(UIViewAnimation)

...............

@end//UIViewAnimation

@interface UIView(UIViewAnimationWithBlocks)

........................

@end//UIViewAnimationWithBlocks

2.2 委托和类别

委托是类别的一种应用:被发送给委托对象的方法可以声明为一个NSObject的类别。创建NSObject的类别称为“创建一个非正式协议”。非正式协议只是一种表达方式,它表示“这里有一些你可能希望实现的方法,当然你也可以不实现这些方法”。示例NSNetService委托方法的部分声明如下:

@interface NSObject(NSNetServiceBrowserDelegateMethods)

- (void)netServiceBrowserWillSearch: (NSNetServiceBrowser*) browser;

- (void)netServiceBrowser: (NSNetServiceBrowser*) aNetServiceBrowser

didFindService: (NSNetService *) service

moreComing: (BOOL) moreComing;

- (void)netServiceBrowserDidStopSearch: (NSNetServiceBrowser*) browser;

- (void)netServiceBrowser: (NSNetServiceBrowser*) aNetServiceBrowser

didRemoveService: (NSNetService *) service

moreComing: (BOOL) moreComing;

@end//NSNetServiceBrowserDelegateMethods

2.2.1 如何知道委托对象能否处理发送给它的消息(响应选择器)

NSNetServiceBrowser首先检查委托对象,通过NSObject类的respondsToSelector:的方法询问其能否响应该选择器。如果委托对象能响应,则NSNetServiceBrowser给委托对象发送消息,反正则不发。

选择器只是一个方法的名称,它以Object-C运行时使用的特殊方式编码,以快速执行查询。使用@selector(方法名称)预编译指令指定选择器。示例:Car类的setEngine:方法的选择器是 @selector(setEngine:)

Car类的setTire: atIndex:方法的选择器是  @selector(setTire: atIndex:)

Car *car = [[Car alloc] init];

if ([car respondsToSelector: @selector(setEngine:)])

{

return YES;

}

3. 类别的局限性

①无法向现有的类中添加新的实例变量(类别没有位置容纳实例变量);

②方法名称冲突,即类别中的新方法的名称与现有类中方法名称重名,类别具有更高的优先级,类别中的方法将完全取代现有类中的方法(再也无法访问现有类中的同名方法)。

二.类扩展Class extensions

类扩展声明格式@interface MyClass(), 可以在类扩展中声明属性和实例变量。

@interface MyClass : NSObject

...................

@end

@interface MyClass()//类扩展

{

float _value;

}

@property(assign,readonly) value;

@end