美文网首页iOS-进阶学习
深入浅出理解分类(category,类别,类目)和类扩展(ext

深入浅出理解分类(category,类别,类目)和类扩展(ext

作者: o本是后山人o偶作前堂客o | 来源:发表于2017-07-26 11:30 被阅读137次

    整理资料.   

    深入浅出理解分类(category,类别,类目)和类扩展(extension,匿名类别)

    苹果推荐分类的使用 ,可以如下使用:

    1.给已经存在的类添加方法

    2.把类的实现分开在几个不同文件中

    减少单个文件的体积

    把不同功能组织到不同的category里

    多个开发者共同完成一个类

    按照需求加载想要的category

    3.声明私有方法

    除此之外,我们还可以转换思想,将category使用到如下地方:

    4.模仿多继承

    5.把framework的私有方法分开



    分类运用场景举例:想要收集每个页面的启动时间。

    问题1:

    项目中已经有上百个页面了,如果一个一个的加,浪费时间不说,以后增加了新页面,还需要添加方法。

    解决方法:

    我们可以发现页面都继承了UIViewController,想要在每个页面都执行的代码,可以写在这些页面的父类中。我们可以把代码写在UIViewController中。

    问题2:

    UIViewController是官方类,我们只能调用期接口,并不能修改他的实现。

    解决方法:

    使用分类(category)。

    1.分类(category)的作用

    1.1作用:可以在不修改原来类的基础上,为一个类扩展方法。

    1.2最主要的用法:给系统自带的类扩展方法。

    2.分类中能写点啥?

    2.1分类中只能添加“方法”,不能增加成员变量。  (在runtime层面可以通过关联对象为分类添加成员属性

    2.2分类中可以访问原来类中的成员变量,但是只能访问@protect和@public形式的变量。如果想要访问本类中的私有变量,分类和子类一样,只能通过方法来访问。

    2.3如果一定要在分类中添加成员变量,可以通过getter,setter手段进行添加,详细以后再写,TODO。

    3.分类的书写方法

    假设我们有一个类,名叫Person,意义是人类。在person类中我们有一个方法,-(void)eat;意义是吃,因为每个人类都会吃。

    人类有很多共同特点,也有很多不同点,比如,并不是每个人都会踢足球。这时候我写一个分类,给喜欢运动的人,这个分类的名字叫做sport。因为是person类的分类,你会看到生成的名字叫做Person+sport。下面让我们来试着添加一下。

    0.分类的使用过程:声明分类->实现分类->使用分类?

    1.在项目中添加类,选择Objective-C File,在?弹出的页面中这样配置。

    2.Person+sport.h头文件

    3.Person+sport.m文件

    4.回到最开始收集页面启动时间的问题,本类是系统的类,这里是UIViewController,我们可以使用分类扩展他的方法,也可以重写他的方法,可以我需要在调用的地方加头文件,所有子类都写头文件和直接在子类写方法没有什么区别,怎么样可以使得不写头文件,子类就能调用我们写的代码呢?

    回答:

    我们可以进行方法交换Method Swizzling(这样可以不必在调用的地方增加头文件),从而使得在实现的时候调用重写的方法。具体实现以后写

    category底层结构

    我们知道,所有OC类和对象,在runtime层都是用struct表示的,category在runtime层使用结构体category_t定义的,包含了如下:

    name 类名

    cls 类

    instanceMethods 对象方法

    classMethods 类方法

    protocols 协议列表

    instanceProperties 添加的所有属性

    4.分类的执行优先级、方法调用顺序  category加载 

    category加载

    在runtime,系统会将包含的内容添加如下:

    把category的实力方法,协议,已经属性添加到类上

    把类方法和协议添加到metaclass

    分类加载和方法调用顺序

    加载:先加载原类的+load方法,再加载分类的+load方法,若多个分类,则根据编译顺序加载

    调用:先调用分类的方法,再调用原类的方法,分类方法会覆盖原类的方法

    分类覆盖类的方法

    如果分类中的方法与类的方法同名,则将会覆盖类的方法,实现原理:分类的对象方法会添加到类的方法列表中,分类的方法会添加到类的相应方法列表中(对象方法添加到类的方法列表,类方法会添加到metaclass列表),而且排列顺序会与原类的同名方法排在一起,且在原类的同名方法的前面。方法调用时会根据顺序先找到第一同名方法就调用,因此会覆盖原类的同名方法。

    4.1在本类和分类有相同的方法时,优先调用分类的方法再调用本类的方法。

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

    5.分类(category)和类扩展(extension)的关系

    1.类扩展(extension)是category的一个特例,有时候也被称为匿名分类。他的作用是为一个类添加一些私有的成员变量和方法。

    2.类扩展能写点啥?和分类不同,类扩展即可以声明成员变量又可以声明方法。

    3.类扩展听上去很复杂,但其实我们很早就认识他了。你记得继承自UIViewController的ViewController和继承自NSObject的类有什么不同么?

    4.继承自UIViewController的ViewController类

    5.对比看一下继承自NSObject的类我们会发现没有上面的代码块

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

    @end

    6.类扩展可以定义在.m文件中,这种扩展方式中定义的变量都是私有的,也可以定义在.h文件中,这样定义的代码就是共有的,类扩展在.m文件中声明私有方法是非常好的方式。

    7.类扩展中添加的新方法,一定要实现。categorygory中没有这种限制。

    category和extension区别1

    category: 类别,分类

    专门用来给类添加新的方法

    不能给类添加成员属性,添加了成员变量,也无法取到

    注意:其实可以通过runtime给分类添加属性

    分类中用@property定义变量,只会生成变量的getter,setter方法的声明,不能生成方法实现和带下划线的成员变量。

    extension:类扩展

    可以说成是特殊的分类,也称作匿名分类

    可以给类添加成员属性,但是是私有变量

    可以给类添加方法,也是私有方法

    category和extension区别2

    虽然有人说extension是一个特殊的category,也有人将extension叫做匿名分类,但是其实两者差别很大。

    extension

    在编译器决议,是类的一部分,在编译器和头文件的@interface和实现文件里的@implement一起形成了一个完整的类。

    伴随着类的产生而产生,也随着类的消失而消失。

    extension一般用来隐藏类的私有消息,你必须有一个类的源码才能添加一个类的extension,所以对于系统一些类,如nsstring,就无法添加类扩展

    category

    是运行期决议的

    类扩展可以添加实例变量,分类不能添加实例变量

    原因:因为在运行期,对象的内存布局已经确定,如果添加实例变量会破坏类的内部布局,这对编译性语言是灾难性的。

    为什么Category只能为对象添加方法,却不能添加成员变量?

    Category只能为对象添加方法,却不能添加成员变量的原因:如果可以添加成员变量,添加的成员变量没有办法初始化

    引用资料:1 http://www.cocoachina.com/ios/20161018/17784.html

                         2 Category分类和Extension类扩展的详解与区别

    相关文章

      网友评论

        本文标题:深入浅出理解分类(category,类别,类目)和类扩展(ext

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