当前位置:首页 > 技术知识 > 正文内容

Objective-C :Category(category什么意思)

maynowei7个月前 (08-02)技术知识128

Category 引入

在日常的开发中,可能会碰到这样的需求:给某个类增加方法。比如说,需要给NSString类增加一个打印的方法。当然,我们可以新建一个类比如TestString,并继承NSString类,在新的类TestString中实现 displayString方法。但是,这种方法有一个明显的缺陷是:只有 TestString类有该方法,NSString类的其他子类,比如 NSMutableString 不能使用该方法。能否给 NSString 类增加一个方法,让NSString以及NSString的所有子类都可以使用呢?答案可以的,Category可以完美的解决这个问题。

Category 的使用

Objective-C 中使用Category的语法是使用 @interface关键字,和定义一个标准的类非常类似,不过不是使用冒号(:,:是继承一个类时使用),而是使用 ,如下:

@interface NSString (PlayString)

- (void)playString:(NSString *)content;

@end

其中:括号内的 PlayStirng 是Category的名称。

可以为任何一个类增加 Category,即使看不到这个类的源代码。当为一个类增加Cateogry后,这个类以及这个类的所有子类都可以使用Category中的方法。在运行时,Category 中的方法和类中原来的代码是没有区别的。比如说,上例中,NSString 类增加Cateogry,Category中定义了 playString方法,该方法的实现如下:

- (void)playString:(NSString *)content
{
    NSLog(@"the content is %@",content);
}

这样,NSString类的实例对象,以及NSString类的所有子类的实例对象,都可以使用 playString方法。如下:

NSString *myString = @"this is original NSString";
[myString playString:myString];
    
NSMutableString *mutString = [[NSMutableString alloc] init];
[mutString appendString:@"this is a subclass of NSString"];
[mutString playString:mutString];

除了给一个类增加方法外,Category 还有以下两种使用场景:

1:将一个大的、复杂的类文件拆分成几个小的类文件。

2:多个人开发同一个类文件时,可以使用Category,分别开发自己的功能。

使用Category的注意事项

1:Category中方法的命名。

(1):尽量不要和原始类中的方法重名,尽管这样是合法的,但是和原始类中的方法重名绝对不是一个好的编程习惯。因为这样造成的后果是,无论是原始类,还是原始类的子类,都无法使用原始类中的那个方法。通常来说,想要覆盖父类中某个方法的情况,更适合用继承来实现,而不是Category。

(2):当一个原始类有多个 Category 时,各个Category 中的方法名要保持相异。尽管多个Category中方法名重复不会提醒错误,但是会发生一些莫名其妙的错误。多个Category中的方法名重复时,每个Category都会向原始类中增加一个函数,这样在运行时,所调用的方法和我们所期望的可能会不一致。这种情况下,具体调用哪个Category中的方法和编译器是相关的。

2:Category中不能增加实例变量。虽然在Category中可以增加属性,但是在 .m文件中,编译器不会自动合成实例变量,以及访问实例变量的 getter/setter 方法。想要为某个原始类增加实例变量,这种情况可以用继承来实现。

Category 原理初探

实际上,Objective-C 中的类经过编译后,在内存中都有一个方法列表,方法列表指向的是该方法的代码块地址。当向某个方法发送消息时,就从方法列表中寻找方法。举例来说有一个类 Person,该类经过编译后生成的方法列表是: setName、getName、getSex ……。现在该类增加一个Category,Category中也实现了方法getName,则再次经过编译后,生成的方法列表是: setName、getName(Category)、getName(原始类)、getSex ……,当给getName方法发送消息时,从类的方法列表中寻找,找到第一个getName方法时,就不在继续往下寻找,这样使用的永远是 Category中实现的 getName 方法。这也是为何要注意Category中方法命名的原因。

相关文章

Flutter 之 ListView(flutter框架)

在 Flutter 中,ListView 可以沿一个方向(垂直或水平方向)来排列其所有子 Widget,常被用于需要展示一组连续视图元素的场景ListView 构造方法ListView:仅适用于列表中...

如何优雅地使用嵌入式事件标志组?

事件标志组嵌入式事件标志组是一种在嵌入式系统中广泛使用的同步机制,主要用于实现多任务间的同步与通信。事件标志组是一组事件标志位的集合,每个位代表一个事件是否发生。它允许任务等待特定的事件发生,当事件发...

Go 语言中的 RWMutex 源码解析、使用场景及应用分析

Go 语言标准库的 sync 包提供了一些非常重要的并发原语来帮助程序开发者处理并发任务。本文将详细解析 sync 包中的 RWMutex,即读写互斥锁(Reader/Writer Mutex),并结...

C++ 原子操作与锁的深度解析:为什么原子操作并非万金油?

大噶好,我是henry,今天来和大家浅浅聊一下为啥C++原子操作并非万能钥匙,原因有三,且听我娓娓道来:一、原子操作的线程安全性C++11 的 std::atomic 确实为单个变量的线程安全操作提供...

C语言编写多线程,什么时候要使用互斥锁?为什么要使用互斥锁?

在多线程编程中,当多个线程同时访问共享资源(如变量、文件等)时,会出现竞态条件(Race Condition)问题,导致程序的行为不可预测。为了避免这种问题,需要使用互斥锁来保护共享资源的访问。互斥锁...

不需安装oracleclient连接oracle数据库方案

在Oracle官方发布ODP.net之前,我们通常使用微软的System.data.OracleClient进行Oracle数据库操作,它的缺点是必须要装Oracle客户端OracleClient,如...