iOS分类(category)、类扩展(extension)、继承的区别

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了iOS分类(category)、类扩展(extension)、继承的区别相关的知识,希望对你有一定的参考价值。

参考技术A ios在2.0就已经推出分类(Category),它允许开发者在不改动原有类的情况下,对该类进行扩展使用。分类(Category)是OC中的特有语法,它是表示一个指向分类的结构体的指针。原则上它只能增加方法,不能增加成员(实例)变量,具体原因看源码:

从中我们可以看出,这个结构体主要包含了分类定义的实例方法与类方法,其中 instance_methods 列表是 objc_class 中方法列表的一个子集,而 class_methods 列表是元类方法列表的一个子集。但这个结构体里面根本没有属性列表。

对于继承,应该都是很熟悉了,网上详细的资料也是非常多,在这里就不再赘述,主要介绍它与分类的区别。

分类(类别/Category)与 类扩展(Extension)

一、分类(类别/Category)

1、适用范围

     当你已经封装好了一个类(也可能是系统类、第三方库),不想在改动这个类了,可是随着程序功能的增加需要在类中增加一个方法,这时我们不必修改主类,只需要给你原来的类增加一个分类。
     将一个大型的类拆分成不同的分类,在不同分类中实现类别声明的方法,这样可以将一个类的实现写到多个.m文件中,方便管理和协同开发。
     分类中的方法可以只声明,不实现,所以在协议不支持可选方法的时候(协议现在已经支持可选方法),通常把分类作为非正式协议使用。
 
2、语法格式
     文件中的语法
@interface 主类类名(分类类名)
//不可以定义成员属性
@end

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

@end

 

     文件名通常为:主类名+分类名

     调用方法时,只需要向主类引用发送消息即可。
 
3、注意事项
  • 分类中方法的优先级比原来类中的方法高,也就是说,在分类中重写了原来类中的方法,那么分类中的方法会覆盖原来类中的方法
  • 分类中只能声明方法,不能添加属性变量,在运行时分类中的方法与主类中的方法没有区别
  • 通常来讲,分类定义在.h文件中,但也可以定义.m文件中,此时分类的方法就变成私有方法
4、如何使用
     定义PYJViewController类的分类
     “PYJViewController+CategoryController.h”文件:
@interface PYJViewController (CategoryController)
- (void)test;
@end

    “PYJViewController+CategoryController.m”文件:

@implementation PYJViewController (CategoryController)
- (void)test {
     NSLog(@"这是一个分类");
}
@end

 

5、虽然不能在分类(类别)中定义成员属性,但是有办法也可以让它支持添加属性和成员变量

    Category是Objective-C中常用的语法特性,通过它可以很方便的为已有的类来添加函数。但是Category不允许为已有的类添加新的属性或者成员变量。     
    一种常见的办法是通过runtime.h中objc_getAssociatedObject / objc_setAssociatedObject来访问和生成关联对象。通过这种方法来模拟生成属性。

    “NSObject+SpecialName.h”文件:

@interface NSObject (SpecialName)
@property (nonatomic, copy) NSString *specialName;
@end

    “NSObject+SpecialName.m”文件:

#import "NSObject+Extension.h"
#import <objc/runtime.h>
static const void *SpecialNameKey = &SpecialNameKey;    
@implementation NSObject (SpecialName)
@dynamic specialName;

- (NSString *)specialName {
    //如果属性值是非id类型,可以通过属性值先构造OC的id对象,再通过对象获取非id类型属性
    return objc_getAssociatedObject(self, SpecialNameKey);
}

- (void)setSpecialName:(NSString *)specialName{
    //如果属性值是非id类型,可以通过属性值先构造OC的id对象,再通过对象获取非id类型属性
    objc_setAssociatedObject(self, SpecialNameKey, specialName, OBJC_ASSOCIATION_RETAIN_NONATOMIC); 
} 

@end

 

 

 

二、类扩展

1、适用范围

     扩展是分类的一种特殊形式。
 
2、语法格式
     @interface 主类类名()
     @end
     扩展通常定义在主类.m文件中,扩展中声明的方法直接在主类.m文件中实现。
 
3、注意事项
  • 扩展中可以声明实例变量,可以声明属性
  • 因为扩展通常定义在主类的.m文件中,所以扩展声明的方法和属性通常是私有的
4、分类和扩展的区别
     分类是不可以声明实例变量,通常是公开的,文件名是:主类名+分类名.h
     扩展是可以声明实例变量,是私有的,文件名为:主类名_扩展标识.h,在主类的.m文件中#import该头文件
 
5.如何使用
    定义 PYJViewController类的扩展
    方式1、以单独的文件定义
    “PYJViewController_ExtensionController.h”文件:
#import"PYJViewController.h"

@interface PYJViewController ()
@property(nonatomic, copy)NSString *stringExtension;
- (void)testExtension;
@end

 

    方式2、在主类的.m文件中定义

    “PYJViewController.m”文件:

#import"PYJViewController.h"

@interface PYJViewController ()
@property(nonatomic, copy)NSString *stringExtension;
- (void)testExtension;
@end

    在主类的.m文件中实现扩展定的方法:

@implementation PYJViewController
- (void)testExtension {
   self.stringExtension = @"给扩展里面定义的属性字符串赋值";
   NSLog(@"定义的属性String是:%@", self.stringExtension);
}
@end

 

 

以上是关于iOS分类(category)、类扩展(extension)、继承的区别的主要内容,如果未能解决你的问题,请参考以下文章

iOS之分类(category)

iOS中分类(Category)、扩展(Extention)和继承(Inheritence)的区别?

iOS - 类别和扩展(Categories和Extensions)

iOS面试 --Objective-C相关

iOS中的分类和扩展

分类(类别/Category)与 类扩展(Extension)