天天看点

objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置

通过分类(category)可以以模块的方式向现有的类添加方法。

它提供了一种简单的方式, 用它可以将类的定义模块化到相关方法的组或分类中。它还提供了扩展现有类定义的简便方式,并且不必访问类的源代码,也无需创建子类。 

objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置
objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置

对于这段代码来说,如何在不改变原来类模型的前提下,给person类扩充一些方法?

有2种方式

继承,需要生成一个新类。

分类(category),这样就不用使用继承,不用生成新类,分类依赖于已经存在的类。开发中很常用。不用修改原来类的代码。

objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置
objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置

在 xcode 里,可以自动生成分类,不用手写,新建 file,然后 oc-file, 分类选项即可。

好处

一个庞大的类可以分模块开发

一个庞大的类可以由多个人来编写,更有利于团队合作

一般以模块命名分类(当然使用作者命名也可以)

objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置
objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置
objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置

分类的作用:

在不改变原来类内容的基础上,可以为类增加一些方法

使用注意:

 1.分类只能增加方法,不能增加成员变量,如果实在想增加新的成员变量,那么可以通过继承实现。

 2.分类方法 实现 中 可以访问原来类中声明的成员变量,否则增加的新方法就没有意义了。

 3.分类可以重新实现原来类中的方法,但是会覆盖掉原来的方法,会导致原来的方法没法再使用(不建议覆盖)

 4.方法调用优先级:分类的方法 --> 原来类的方法  --> 父类的方法

优先去分类中查找,然后再去原来类中找,最后再去父类中找,如果有多个分类,且多个分类的里面的方法都覆盖了原类方法,那么调用顺序和编译顺序有关。看编译器先编译哪个文件,就先调用哪个文件的覆盖的方法。

单击—项目——出现build phases ——中的 compile source 中查看:

编译的文件顺序,顺序可以人为改变。还发现,头文件是不被编译的,这就再次验证,头文件打酱油的特点,只是为了编程规范,所以之前,不写声明,直接写类的实现,不会报错的原因就是这样。

给系统自带的类添加分类

实际开发中,常常给系统自带的类添加分类,(系统自带的类无法修改,但是可以添加分类增加方法)不一定只是给自定义的类添加分类

这里再次说明:分类的名称,最好是按照模块命名(或者功能命名),让人一目了然。

给nsstring增加一个类方法:计算某个字符串中数字的个数,再增加一个对象方法:计算当前字符串中数字的个数

objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置
objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置

unichar 本质是:无符号短整型(有时候 char 类型当做 整数类型)

typedef unsigned short unichar;

在类方法里,+ (int)numbercountofstring:(nsstring *)str;

我们可以不用写那么多代码来实现,而是依靠对象方法来实现,直接return [str numbercount];即可!

再次注意

category可以访问原始类的实例变量,但不能添加变量,只能添加方法。如果想添加变量,可以考虑通过继承创建子类

category可以实现原始类的方法,但不推荐这么做,因为它是直接替换掉原来的方法,这么做的后果是再也不能访问原来的方法

多个category中如果实现了相同的方法,只有最后一个参与编译的才会有效

其实类也是一个对象,是class类型的对象,简称“类对象”,class类型的定义:

typedef struct objc_class *class;

类名就代表着类对象,每个类只有一个类对象,即,内存中只有一份类对象

+load 和 +initialize方法

+load方法:

在程序启动的时候会加载所有的类和分类,并调用所有类和分类的+load方法。先加载父类,再加载子类。也就是先调用父类的+load方法,再调用子类的+load方法。先加载原始类,再加载原始类的分类 。不管程序运行过程有没有用到这个类,都会调用+load加载 。

+initialize方法:

在第一次使用某个类时(比如创建对象等),就会调用一次+initialize方法,一个类只会调用一次+initialize方法,先调用父类的,再调用子类的。先加载原类,再加载分类

description方法

-description方法

使用nslog和%@输出某个对象时,会调用-description方法,并拿到返回值进行输出,没有声明在 nsobject 头文件。

+ description方法

使用nslog和%@输出某个类时,会调用+description方法,并拿到返回值进行输出,在 nsobject 类的头文件里

修改nslog的默认输出

重写-description或者+description方法即可

死循环陷阱

如果在-description方法中使用nslog打印self

objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置
objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置

每个类的方法列表都存储在类对象中

每个方法都有一个与之对应的sel类型的数据

根据一个sel数据就可以找到方法的地址,进而调用方法

sel类型的定义

sel其实是对方法的一种包装,将方法包装成一个sel类型的数据,去找对应的方法地址。找到方法地址就可以调用方法,其实发送的消息就是sel

objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置
objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置
objective-c 语法快速过(5)oc 的分类-Category类的本质SEL数据类型方法的存储位置

sel对象的创建

sel对象的其他用法

// 将sel对象转为nsstring对象

person *p = [person new];

// 调用对象p的test方法

[p performselector:@selector(test)];

nslog输出增强

__file__ :源代码文件名

__line__ :nslog代码在第几行

_cmd :代表着当前方法的sel

辛苦的劳动,转载请注明出处,谢谢……

http://www.cnblogs.com/kubixuesheng/p/4314169.html