美文网首页
iOS开发阅读 - Objective-C系列(2)

iOS开发阅读 - Objective-C系列(2)

作者: 夕醒 | 来源:发表于2018-05-01 14:12 被阅读16次

iOS开发阅读 - Objective-C系列(目录)

分类(Category)和类扩展(Extension)

什么是分类,什么是类扩展,他们分别用在什么场合,各有哪些不能被替代的特点?

分类(Category):它是在运行时决议的,我们可以给类或者系统类添加实例方法方法。我们添加的实例方法,会被动态的添加到类结构里面的methodList列表里面。
类扩展(Extension):看起来很像一个匿名的category,但是extension和有名字的category几乎完全是两个东西。
extension在编译时决议,它就是类的一部分,在编译期和头文件里的@interface以及实现文件里的@implement一起形成一个完整的类,它伴随类的产生而产生,亦随之一起消亡。extension一般用来隐藏类的私有信息,你必须有一个类的源码才能为一个类添加extension,所以你无法为系统的类比如NSString添加extension。

分类用到的场合

1.可以为类添加方法
2.在runtime可以为类添加属性
在1这样一种情况大家用的应该比较多,可以在不修改原来类的基础上,为一个类扩展方法。最主要的用法,给系统自带的类扩展方法。
比方说,在工程里每个UIViewController里都想要做同样一件事情,如果一个一个的加,浪费时间不说,以后增加了新页面,还需要添加方法。
这就可以用到分类,想要在每个页面都执行的代码,可以写在这些页面的父类中。我们可以把代码写在UIViewController中。
但是它也有局限性:
1.分类中只能添加方法,不能增加成员变量。(可以添加属性)
2.分类中可以访问原来类中的成员变量,但是只能访问@protect和@public形式的变量。如果想要访问本类中的私有变量,分类和子类一样,只能通过方法来访问。

为什么不能增加成员变量?

objc_class结构体中,ivarsobjc_ivar_list(成员变量列表)指针;methodLists是指向objc_method_list指针的指针。在Runtime中,objc_class结构体大小是固定的,不可能往这个结构体中添加数据,只能修改。所以ivars指向的是一个固定区域,只能修改成员变量值,不能增加成员变量个数。methodList是一个二维数组,所以可以修改*methodLists的值来增加成员方法,虽没办法扩展methodLists指向的内存区域,却可以改变这个内存区域的值(存储的是指针)。
简单的我想讲一下如何实现为分类添加属性,但是不会生成_变量(带下划线的成员变量),也不会生成添加属性的getter和setter方法的实现:
比如,为UIViewController添加显示的时间这样一个属性。

static char *TimeKey = "TimeKey";

-(void)setTime:(NSTimeInterval)time {
    objc_setAssociatedObject(self, TimeKey, cloudox, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

-(NSString *)time{
    return objc_getAssociatedObject(self, TimeKey);
}

注意,要导入头文件。

#import <objc/runtime.h>

分类的执行优先级

1.在本类和分类有相同的方法时,优先调用分类的方法再调用本类的方法。
2.如果有两个分类,他们都实现了相同的方法,如何判断谁先执行?分类执行顺序可以通过targets,Build Phases,Complie Source进行调节,注意执行顺序是从上到下的。(只有两个相同方法名的分类)

extension

类扩展(extension)是category的一个特例,有时候也被称为匿名分类。他的作用是为一个类添加一些私有的成员变量和方法。
类扩展能写点啥?和分类不同,类扩展即可以声明成员变量又可以声明方法。
类扩展听上去很复杂,但其实我们很早就认识他了。

#import <UIKit>
#import "LoginViewController.h"

@interface LoginViewController ()
@property (nonatomic, strong) IBOutlet UITextField* nameTF;
@property (nonatomic, strong) IBOutlet UITextField* pwdTF;
- (IBAction)login:(id)sender;
@end

@implementation LoginViewController

- (void)viewDidLoad {
  [super viewDidLoad];
}

- (void)didReceiveMemoryWarning {
  [super didReceiveMemoryWarning];
}

- (IBAction)login:(id)sender {

}
@end

是不是很熟悉了,2333。
对比看一下继承自NSObject的类我们会发现没有上面的代码块。

 @interface LoginViewController()//这就是类扩展的写法
 @end

类扩展可以定义在.m文件中,这种扩展方式中定义的变量都是私有的,也可以定义在.h文件中,这样定义的代码就是共有的,类扩展在.m文件中声明私有方法是非常好的方式。
类扩展中添加的新方法,一定要实现。category中没有这种限制。

比较category和extension, 总结来自于(VV木公子 简书作者):
extension在编译期决议,它就是类的一部分,但是category则完全不一样,它是在运行期决议的。extension在编译期和头文件里的@interface以及实现文件里的@implement一起形成一个完整的类,它、extension伴随类的产生而产生,亦随之一起消亡。
extension一般用来隐藏类的私有信息,你必须有一个类的源码才能为一个类添加extension,所以你无法为系统的类比如NSString添加extension,除非创建子类再添加extension。而category不需要有类的源码,我们可以给系统提供的类添加category。
extension可以添加实例变量,而category不可以。
extension和category都可以添加属性,但是category的属性不能生成成员变量和getter、setter方法的实现。

额外的补充,Objective-C中的点语法说明:

如果点表达式出现在 "=" 左边,该属性名称的setter方法将被调用。如果点表达式出现在右边,该属性名称的getter方法将被调用。所以在oc中点表达式其实就是调用对象的settergetter方法的一种快捷方式。

@synthesize 还有一个作用,可以指定与属性对应的实例变量,例如@synthesize var = xxxx;那么self.var其实是操作的实例变量而是xxxx,不是_var了。

相关文章

网友评论

      本文标题:iOS开发阅读 - Objective-C系列(2)

      本文链接:https://www.haomeiwen.com/subject/jhuelftx.html