Category在我目前的工作经验中用的不算特别多,但确实也是一个非常方便整洁的方法。
一下几个要点都是个根据书上稍加翻译和理解得来,不足之处还望指出改正。
使用Category来分割类的实现
一个大类通常会因为内含有多种方法而变得臃肿复杂,相对于使用Category而言,没有更好的办法去做分割这样的重构工作了。
Category能将一个大类中的方法按逻辑进行分割成不同的分类,不仅有利于开发,debug起来也比较清晰。
比方有一个Person类,里面有非常多的,不同逻辑的方法:
#import <Foundation/Foundation.h>
@interface Person : NSObject<NSCopying>
@property (nonatomic, copy, readonly) NSString *firstName;
@property (nonatomic, copy, readonly) NSString *lastName;
@property (nonatomic, copy, readonly) NSArray *friends;
- (instancetype)initWithFirstName:(NSString *)firstName lastName:(NSString *)lastName;
#pragma - mark Friendship Method
- (void)addFriend:(Person *)aFriend;
- (void)removeFriend:(Person *)aFriend;
#pragma - mark Work Method
- (void)performDaysWork;
- (void)takeVacationFromWork;
#pragma - mark Play Method
- (void)goToCinema;
- (void)goToPlaySoccer;
@end
在这样的情况下,Person就会包含一长串的方法。采用Category“分割”后如下:
#import <Foundation/Foundation.h>
@interface Person : NSObject<NSCopying>
@property (nonatomic, copy, readonly) NSString *firstName;
@property (nonatomic, copy, readonly) NSString *lastName;
@property (nonatomic, copy, readonly) NSArray *friends;
- (instancetype)initWithFirstName:(NSString *)firstName lastName:(NSString *)lastName;
@end
@interface Person(Friendship)
- (void)addFriend:(Person *)aFriend;
- (void)removeFriend:(Person *)aFriend;
@end
@interface Person(Work)
- (void)performDaysWork;
- (void)takeVacationFromWork;
@end
@interface Person(Play)
- (void)goToCinema;
- (void)goToPlaySoccer;
@end
这样以来,不同逻辑的方法有了更明显的区分,不过随着方法的变多,文件本身也会变得臃肿,还有一种方法就是将大类分割成独立的分类文件,建分类文件的方法就不多说了:
- command + N
- 选择Objective-C File
- File Type 为Category ,正确选择父类
- File一栏只需要填后缀名即可(比如Friendship)
建完后目录里应该会多出
Person + Friendship.h和Person + Friendship.m
两个文件,然后按照你的分类方式声明方法与实现即可。
在开发一个库的时候,写一些私有的Category也是不错的方法,这样可以避免暴露一些不必要暴露的API。
给你的分类方法加上合适的前缀
Category通常用于给一个第三方的类,你不知道源码的类来添加使用的。这是一个非常有用的功能,但是容易被忽视的地方在于方法的命名上面。
当分类被加载的时候,runtime会逐个将分类中的方法添加到原有类的方法表中(之前文章提到过,类的方法都是以一张表的形式存在)。如果这个同样名字的方法在类中早已经有了实现,那么分类中的方法的实现将会替换掉原来类中的实现方法。这个覆盖的过程是可以重复的。如果当有第二个Category也写了一个方法是同样的方法名,那么当第二个Category导入的时候,方法会被再覆盖一次。
解决这样的问题就是需要在你的Category方法中加上前缀:
#import <Foundation/Foundation.h>
@interface NSString (MyString)
- (NSString *)fh_upperCaseString;
- (NSString *)fh_lowerCaseString;
@end
理论上来说,你并不需要为你的Category本上加上一个前缀,因为这样编译器也不会报错,只不过有一个重复定义的警告,还是不要这样做。
虽然说防止Category重复声明方法是不可能完全避免的,但是通过增加前缀的方式我们可以大大降低发生的概率。不这样的话,产生的后果是非常难Debug的。
Category还有很多内容,下回再做整理学习。