一、分类-Category

1.思考:如何在不改变原来类的模型的前提下,扩展分类的方法?

两种方式:

1》继承:子类可继承父类原有的方法,并且增加自己的方法.

2》分类;  可以给某个类扩 充方法(不修改原来的代码)

2.分类的使用格式

1》分类的声明

@interface 类名 (分类名称)

// 方法声明

@end

2》分类的实现

@implementation类名 (分类名称)

// 方法实现

@end

分类使用示例:

给NSString类添加一个类方法,计算某字符串中阿拉伯数字的个数

#import <Foundation/Foundation.h>@interface NSString (number)
/* 给NSString类添加一个类方法,计算某字符串中阿拉伯数字的个数 */
+ (int)numberCountOfString:(NSString *)str;
/* 对象方法 自己的数字个数不用参数了*/
- (int)numberCount;
@end

代码测试:

int main()
{// 扩展类方法的调用int count1 = [NSString numberCountOfString:@"1fh2hg3 h456d78"];NSLog(@"1fh2hg3 h456d78 has %d num ",count1);// 分析:这个方法不好还得用类的形式调用,改进用对象方法// 扩展对象方法的调用int count2 = [@"123asdfg456" numberCount];NSLog(@"123asdfg456 has %d num ",count2);return 0;
}

2015-03-23 14:53:39.922 catgory分类[1042:96183] 1fh2hg3 h456d78 has 8 num 

2015-03-23 14:53:39.922 catgory分类[1042:96183] 123asdfg456 has 6 num 

分类总结:

分类作用:
 再不改变原来类内容基础上增加一些方法;
好处:
一个庞大的类 可以分模块开发
一个庞大的类可以有多个人编写,有利于团队合作
注意:
1. 分类不能扩充成员变量:
 2. 分类方法实现中可以访问成员变量
 3. 分类可以重新实现原来类中的方法,但会覆盖掉原来的方法,导致原来的方法中没法使用
 4. 方法调用优先级:分类(后编译的)-》原来类-》父类
 
还有一个问题,多个分类同时重新实现一个方法那么调用哪个呢?
哪个.m后编译,就会覆盖掉之前编译的,可在xcode中调整。

二 .类的本质

类本身也是一个对象,是个class类型的对象,简称类对像,他只在内存中存在一份。

Class类型的定义  typedefstructobjc_class *Class;

类名就代表着类对象,每个类只有一个类对象。

2.通过代码类理解类的本质

Person类的声明,声明了一个类方法

#import <Foundation/Foundation.h>@interface Person : NSObject
{int _age;
}
@property int age;+ (void)test;@end

Person类的实现

#import "Person.h"@implementation Person
+ (void)test
{NSLog(@"调用了类的类方法test方法");
}@end

主函数:

int main()
{/* p1 p2 都指向创建的Person类的对象*/Person *p1 = [[Person alloc] init];Person *p2 = [[Person alloc] init];/* 获取内存中类对象的两种方式 都返回Person这个类在内存中的地址 */// 1.调用对象的class 方法Class c = [p1 class];Class c2 = [p2 class];// 2.调用类的class方法Class c3 = [Person class];NSLog(@"c = %p c2 = %p c3 =%p",c,c2,c3);// 结果c1 = c2 = c3 说明同一个类在内存中只有一个/* 类名就是类对象 和class返回的类对象是一样的 */[Person test];[c3 test];// 类对象 = 类名 也可以创建对象Person *p3 = [[c alloc] init];p3.age = 100;NSLog(@"p3 age %d",p3.age);return 0;
}

2015-03-23 15:22:22.457类的本质[1109:104403] c = 0x1000025d8 c2 = 0x1000025d8 c3 =0x1000025d8

2015-03-23 15:22:22.457类的本质[1109:104403]调用了类的类方法test方法

2015-03-23 15:22:22.458类的本质[1109:104403]调用了类的类方法test方法

2015-03-23 15:22:22.458类的本质[1109:104403] p3 age 100

三 .    +load和+initialize方法

+load
1. 程序启动时就会加载项目中所有的类和分类,并调用每个类和分类的+load方法,  只会调用一次
2. 先加载父类在加载子类最后加载分类,也就是先调用父类的+load方法,在调用子类的+load方法,最后加载分类的+  load方法,
3.不管程序运行过程中有没有用到这个类,都会加载,且只加载一次
   
+initialize
1.当第一次使用某个类时(比如创建对象),就会调用当前类的initialize方法
2.先初始化父类在初始化子类,先调用父类的initialize方法在调用子类的initalize方法,只会初始化一次;
  分类也会加载,但是用的时候优先使用分类的initialize方法不会加载原类的initlize方法。
3.使用了initalize方法就可以监听什么时候使用了类

+load和+initialize方法的调用过程代码示例:

先说明一下用到的几个类的关系:
Person类继承于OBJect类,并且又增加了分类,Student类继承于Person类。
Person类的声明同上文:
Person类的实现:
#import "Person.h"@implementation Person
+ (void)test
{NSLog(@"调用了类的类方法test方法");
}
/* 重写了load方法:只要程序运行就会加载一次仅此一次 */
+ (void)load
{NSLog(@"Person load!!");
}
/* 重写了initialize方法:只在用到的时候会调用*/
+ (void)initialize
{NSLog(@"Person initialize!!");
}
@end

Person类的分类声明:

#import "Person.h"@interface Person (loadTest)@end

Person类的分类的实现:

#import "Person+loadTest.h"@implementation Person (loadTest)
/* 重写了load方法:只要程序运行就会加载一次仅此一次 */
+ (void)load
{NSLog(@"Person+loadTest load!!");
}
/* 重写了initialize方法:只在用到的时候会调用*/
+ (void)initialize
{NSLog(@"Person+loadTest initialize!!");
}
@end

Student类的声明:

#import "Person.h"
/* Student类的声明 继承于Person类*/
@interface Student : Person@end

Student类的实现:

#import "Student.h"@implementation Student
/* 重写了load方法:只要程序运行就会加载一次仅此一次 */+ (void)load
{NSLog(@"Student load!!");
}
/* 重写了initialize方法:只在用到的时候会调用*/
+ (void)initialize
{NSLog(@"Student initialize!!");
}
@end

+load和+initialize方法的调用过程测试代码:

1.什么也不执行,查看调用的过程?
int main()
{return 0;
}

2015-03-23 15:42:40.415类的本质[1246:110623] Person load!!

2015-03-23 15:42:40.417类的本质[1246:110623] Student load!!

2015-03-23 15:42:40.417类的本质[1246:110623] Person+loadTest load!!

从打印的结果可见:

不管程序运行过程有没有用到这个类,都会调用+load加载进内存,加载的顺序是,先加载父类在加载子类最后加载分类,只加载一次。

2.程序运行过程中只使用Person这个类,查看调用的过程?

int main()
{// 只是用Person这个类[[Person alloc] init];return 0;
}

2015-03-23 15:47:35.384 类的本质[1254:112282] Person load!!

2015-03-23 15:47:35.385 类的本质[1254:112282] Student load!!

2015-03-23 15:47:35.385 类的本质[1254:112282] Person+loadTest load!!

2015-03-23 15:47:35.386 类的本质[1254:112282] Person+loadTest initialize!!

从打印的结果可见:

加载过程同上,当使用了这个Person类时,优先使用分类的initialize方法不会加载原类的initlize方法,由于没使用子类并不会初始化子类。

3.使用子类

int main()
{//程序启动时就会加载项目中所有的类,不管用不用// 使用Student类[[Student alloc] init];return 0;
}

2015-03-23 15:52:42.834 类的本质[1268:113796] Person load!!

2015-03-23 15:52:42.836 类的本质[1268:113796] Student load!!

2015-03-23 15:52:42.836 类的本质[1268:113796] Person+loadTest load!!

2015-03-23 15:52:42.837 类的本质[1268:113796] Person+loadTest initialize!!

2015-03-23 15:52:42.837 类的本质[1268:113796] Student initialize!!

从打印的结果可见:

先初始化父类在初始化子类,先调用父类的initialize方法在调用子类的initalize方法,只会初始化一次;分类也会加载,但是用的时候优先使用分类的initialize方法不会加载原类的initlize方法。

四.SEL数据

1. 方法在内存中的存储:

每个方法在内存中都有一个SEL数据和它对应虚拟成下面的形式

SEL s1 = +test 地址

SEL s2 = -test 地址

2.Objective-C中调用函数的方法是“消息传递”,是利用performSelector传递一个sel类型的消息,再根据sel消息找到方法的地址进而去调用这个方法。

3.  SEL类型的定义

typedef struct objc_selector    *SEL;

4. SEL对象的创建

SEL s = @selector(test);

SEL s2 = NSSelectorFromString(@"test");

SEL使用示例:

Person类的声明
#import <Foundation/Foundation.h>@interface Person : NSObject
/* 类里面包含方法列表 */
// 每个方法在内存中都有一个sel和他对应
+ (void)test1;
- (void)test2;
- (void)test3:(NSString *)str;
@end

Person类的实现:

#import "Person.h"@implementation Person
+ (void)test1
{NSLog(@"test1");
}
- (void)test2
{// _cmd代表本方法// NSStringFromSelector方法接收sel类型的数据,并将其转化为字符串NSString *str = NSStringFromSelector(_cmd);/* 打印本方法名 */NSLog(@"In %@ function",str);
}
- (void)test3:(NSString *)str
{NSLog(@"In test3 传给我的字符串是 %@",str);
}
@end

sel的几种用法:

int main()
{Person *p = [[Person alloc] init];// 直接调用test2方法[p test2];// 使用SEL调用方法的过程// 1.把test2包装成SEL类型的数据// 2.根据SEL数据找到对应的方法地址// 3.根据方法地址调用对应的方法[p performSelector:@selector(test2)];/* 方法名字符串调用方法 */NSString *name = @"test2";// 把字符串变成SELSEL s2 = NSSelectorFromString(name);[p performSelector:s2];/* 带参数的方法调用 */// 方式一[p performSelector:@selector(test3:) withObject:@"hello sel"];// 方式二SEL s3 = @selector(test3:);[p performSelector:s3 withObject:@"123"];return 0;
}

2015-03-23 16:31:30.853 sel[1368:124580] In test2 function

2015-03-23 16:31:30.854 sel[1368:124580] In test2 function

2015-03-23 16:31:30.854 sel[1368:124580] In test2 function

2015-03-23 16:31:30.854 sel[1368:124580] In test3传给我的字符串是 hello sel

2015-03-23 16:31:30.855 sel[1368:124580] In test3传给我的字符串是 123

sel总结;

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

2 每个方法都有一个与之对应的SEL类型的对象

3. 根据一个SEL对象就可以找到方法的地址,进而调用方法

4. 使用调用方法需要performSelector传递一个sel类型消息。
5. _cmd :代表着当前方法的SEL

五.description方法

1. -description方法

使用NSLog和%@输出某个对象时,会调用对象的-description方法,并拿到返回值进行输出

2. + description方法

使用NSLog和%@输出某个类对象时,会调用类对象+description方法,并拿到返回值进行输出

Objective-C基础学习笔记(七)-类的本质与SEL相关推荐

  1. 逆向脱壳破解分析基础学习笔记七 堆栈图(重点)

    本文为本人 大神论坛 逆向破解脱壳学习笔记之一,为本人对以往所学的回顾和总结,可能会有谬误之处,欢迎大家指出. 陆续将不断有笔记放出,希望能对想要入门的萌新有所帮助,一起进步 堆栈图 首先给定一段反汇 ...

  2. Python基础学习笔记-7.类-面向对象的编程

    7.类-面向对象的编程 7.1.类的定义 7.1.1.类的概述 面向对象更符合人类对客观世界的抽象和理解 一切皆对象 一只小狗,一把椅子,一张信用卡,一条巧克力... 一切对象,都有自己内在的属性 狗 ...

  3. Python3零基础学习笔记七

    lambda 表达式(又名匿名函数)     作用:        创建一个匿名函数对象        同def类似,但不提供函数名     语法:        lambda [形参1,形参2... ...

  4. Python3 基础学习笔记 C08 【类】

    CSDN 课程推荐:<8小时Python零基础轻松入门>,讲师齐伟,苏州研途教育科技有限公司CTO,苏州大学应用统计专业硕士生指导委员会委员:已出版<跟老齐学Python:轻松入门& ...

  5. 吴恩达《机器学习》学习笔记七——逻辑回归(二分类)代码

    吴恩达<机器学习>学习笔记七--逻辑回归(二分类)代码 一.无正则项的逻辑回归 1.问题描述 2.导入模块 3.准备数据 4.假设函数 5.代价函数 6.梯度下降 7.拟合参数 8.用训练 ...

  6. Python基础学习笔记之(二)

    Python基础学习笔记之(二) zouxy09@qq.com http://blog.csdn.net/zouxy09 六.包与模块 1.模块module Python中每一个.py脚本定义一个模块 ...

  7. Python基础学习笔记之(一)

    Python基础学习笔记之(一) zouxy09@qq.com http://blog.csdn.net/zouxy09 前段时间参加微软的windows Azure云计算的一个小培训,其中Pytho ...

  8. 多人网络游戏服务器开发基础学习笔记 II: 帧同步 | 游戏客户端预测原理分析 | FPS 游戏状态同步

    这篇是对书本 网络多人游戏架构与编程 的学习第二篇(第一篇:多人网络游戏服务器开发基础学习笔记 I:基本知识 | 游戏设计模式 | 网游服务器层次结构 | 游戏对象序列化 | 游戏 RPC 框架 | ...

  9. 尚学堂JAVA基础学习笔记_2/2

    尚学堂JAVA基础学习笔记_2/2 文章目录 尚学堂JAVA基础学习笔记_2/2 写在前面 第10章 IO技术 1. IO入门 2. IO的API 3. 装饰流 4. IO实战 5. CommonsI ...

最新文章

  1. java6特性_Java6的新特性
  2. Linux多线程同步的几种方式
  3. 面试官:HashSet是如何保证元素不重复的?
  4. OC继承以及实例变量修饰符
  5. 大学生适合学习的软件 网站推荐
  6. html新建文件夹,javascript实现新建文件夹的功能
  7. 大数据时代个人隐私权保护机制构建与完善
  8. Power Query之二 可视化数据处理
  9. HDU 5956 The Elder (树形DP + 斜率优化)
  10. birthday日期类型转化
  11. 关于Android的post,get、cookie网络获取的一些坑
  12. iview upload 上传时携带额外参数
  13. 安卓九宫格上传图片(微信朋友圈,QQ动态)小demo
  14. 腾讯35万年薪难求才 都去了微软和谷歌
  15. 【Proteus】DS18B20简易温控器
  16. 开学了!这些Linux认证你要知道
  17. 通过服务器能查到对方的信息吗,来看!通过微博用户名能查到对方什么信息?...
  18. 为何说特斯拉Model 3有望成为2021年欧洲电动车销冠?
  19. CE认证和FCC认证区别
  20. 模糊逻辑学习--什么是模糊逻辑

热门文章

  1. BUG记录:Exception evaluating SpringEL expression
  2. Blueprints - UE4第三人称游戏角色控制
  3. 从零记录腾讯云建站全过程
  4. 【嵌入式开发】LED 驱动 ( LED发光二极管原理 | 底板原理图分析 | 核心板原理图分析 | GPIO | 裸板程序烧写流程 )
  5. 华为手机android目录,华为手机中的文件夹代表什么,哪些可以删除,看完秒懂...
  6. Mac 搭建http协议的SVN服务
  7. 什么是工业用计算机,工业电脑和普通电脑有什么不同?
  8. Compose Preview 的 UX 设计之旅
  9. fedora 14 安装ati显卡驱动
  10. sap 发送mesage_SAP MESSAGE