【IOS 开发】Objective - C 面向对象高级特性 - 包装类 | 类处理 | 类别 | 扩展 | 协议 | 委托 | 异常处理 | 反射

时间:2021-03-04 03:41:48

一. Objective-C 对象简单处理

1. 包装类

(1) 包装类简介

NSValue 和 NSNumber :

-- 通用包装类 NSValue : NSValue 包装单个 short, int, long, float, char, id, 指针 等数据;

-- NSNumber 包装类 : 用于包装 C 语言数据类型;

NSNumber 方法 :

-- "+ numberWithXxx :" : 将特定类型的值包装成 NSNumber;

-- "- initWithXxx :" : 先创建一个 NSNumber 对象, 再用一个基本类型的值来初始化 NSNumber;

-- "- xxxValue :" : 返回 NSNumber 对象包装的基本类型的值;

(2) 包装类代码示例

代码示例 :

/*************************************************************************
    > File Name: OCNSNumberDemo.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 六 10/ 3 12:50:15 2015
 ************************************************************************/

#import <Foundation/Foundation.h>

int main(int argc, char * argv[])
{
	@autoreleasepool {
		NSNumber * num_int = [NSNumber numberWithInt : 10];
		NSNumber * num_double = [NSNumber numberWithDouble : 10];
		NSNumber * num_char = [NSNumber numberWithChar : 'A'];

		NSLog(@"number_int : %d, number_double : %g, num_char : %c",
			[num_int intValue], [num_double doubleValue], [num_char charValue]);

		NSNumber * num_int1 = [[NSNumber alloc] initWithInt : 10];
		NSNumber * num_double1 = [[NSNumber alloc] initWithDouble : 10];
		NSNumber * num_char1 = [[NSNumber alloc] initWithChar : 'A'];

		NSLog(@"number_int1 : %d, number_double1 : %g, num_char1 : %c",
			[num_int1 intValue], [num_double1 doubleValue], [num_char1 charValue]);
	}
}

-- 执行结果

localhost:oc_object octopus$ clang -fobjc-arc -framework Foundation OCNSNumberDemo.m
localhost:oc_object octopus$ ./a.out
2015-10-03 13:00:46.465 a.out[887:507] number_int : 10, number_double : 10, num_char : A
2015-10-03 13:00:46.468 a.out[887:507] number_int1 : 10, number_double1 : 10, num_char1 : A

2. description 方法

(1) description 方法简介

description 方法简介 : 类似于 Java 中 Object 的 toString() 方法;

-- 方法来源 : description 是 NSObject 中定义的, 所有的方法都有该方法;

-- 默认方法 : description 默认方法返回 <类名: 地址>;

-- 输出对象 : NSLog() 函数输出一个对象, 其实输出的是该对象的 description 方法;

-- 示例 : OCPerson * person, 打印 [person description] 和 person 输出结果是一样的;

(2) description 示例代码

示例代码 :

/*************************************************************************
    > File Name: OCDescriptionDemo.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 六 10/ 3 14:25:28 2015
 ************************************************************************/
#import <Foundation/Foundation.h>

@interface OCDescriptionDemo : NSObject
@property (nonatomic, copy) NSString * name;
@property (nonatomic, assign) int age;
- (id) initWithNameAndAge : (NSString *) set_name setAge : (int) set_age;
@end

@implementation OCDescriptionDemo
@synthesize name;
@synthesize age;
- (id) initWithNameAndAge : (NSString *) set_name setAge : (int) set_age
{
	self.name = set_name;
	self.age = set_age;
	return self;
}
- (NSString *) description
{
	NSString * des = [NSString stringWithFormat :
		@"<OCDescription[name = %@, age = %d]>", self.name, self.age];
	return des;
}
@end

int main(int argc, char * argv[])
{
	@autoreleasepool {
		OCDescriptionDemo * description = [[OCDescriptionDemo alloc] initWithNameAndAge : @"Tom" setAge : 18];
		NSLog(@"%@", description);
	}
}

-- 执行结果

localhost:oc_object octopus$ clang -fobjc-arc -framework Foundation OCDescriptionDemo.m
localhost:oc_object octopus$ ./a.out
2015-10-03 14:50:18.665 a.out[970:507] <OCDescription[name = Tom, age = 18]>

3. == 或 isEqual : 方法

(1) "==" 运算符

"==" 简介 :

-- 作用 : 判断两个变量是否相等;

-- 前提 : 两个变量都是基本类型, 两个变量相等返回 true; 指针类型变量比较地址没有任何意义;

(2) 常量池

常量池 :

-- 作用 : 保证相同的字符串常量至右一个, 不能出现多个相同的副本;

-- 例外 : 使用 [NSString stringWithFormat] 方法创建的字符串不会放入常量池;

(3) isEqual 方法

"isEqual" 方法简介 :

-- 来源 : isEqual 方法是 NSObject 类提供的实例方法, 用于判断相同类型的两个变量是否相等;

-- 默认 : 默认方法还是比较地址, 需要开发者重写这个方法;

-- NSString 的 isEqual 方法 : NSString 的 isEqual 方法是判断两个字符串是否相等, 包含的字符串相同就会返回 true;

-- isEqualToString 方法 : 方法 : NSString 中定义的 isEqualToString 方法用于判断当前字符串 与 另一个字符串的字符串序列是否相等;

重写 isEqual 方法标准 :

-- 自反性 : 对象 x, [x isEqual : x] 必须返回 true;

-- 对称性 : 对象 x 和 y, 如果 [x isEqual : y] 返回值 必须与 [y isEqual : x] 返回值相同;

-- 传递性 : 对象 x , y 和 z, [x isEqual : y] = true, [y isEqual : z] = true, 那么 x z 也相等;

-- 一致性 : x , y 对象无论调用多少次, 返回值结果都应该保持一致;

-- nil 对比 : 如果 x 不是 nil, [x isEqual : nil] 必须返回 false;

(4) "==" 和 "isEqual" 示例源码

示例源码 :

/*************************************************************************
    > File Name: OCEqual.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 六 10/ 3 16:07:56 2015
 ************************************************************************/
#import <Foundation/Foundation.h>

@interface OCEqual : NSObject
@property (nonatomic, copy) NSString * name;
@property (nonatomic, assign) int age;

- (id) initWithName : (NSString *) set_name setAge : (int) set_age;
@end

@implementation OCEqual
@synthesize name;
@synthesize age;

- (id) initWithName : (NSString *) set_name setAge : (int) set_age
{
	self.name = set_name;
	self.age = set_age;
	return self;
}

- (BOOL) isEqual : (id) other
{
	if(self == other)
		return YES;

	if(other != nil && [other isMemberOfClass : OCEqual.class])
	{
		OCEqual * equal = (OCEqual *) other;
		return [self.name isEqual : equal.name] && (self.age == equal.age);
	}
	return NO;
}
@end

int main(int argc, char * argv[])
{
	@autoreleasepool {
		int int_a = 10;
		int double_a = 10.0;

		NSLog(@"int_a == double_a : %d", (int_a == double_a));

		NSString * str_a = @"Octopus";
		NSString * str_b = @"Octopus";
		NSString * str_c = [NSString stringWithFormat : @"Octopus"];
		NSString * str_d = [NSString stringWithFormat : @"Octopus"];

		NSLog(@"str_a == str_b : %d, str_c == str_d : %d, [str_c isEqual : str_d] : %d",
			str_a == str_b, str_c == str_d, [str_c isEqual : str_d]);

		OCEqual * equal_a = [[OCEqual alloc] initWithName : @"Tom" setAge : 18];
		OCEqual * equal_b = [[OCEqual alloc] initWithName : @"Jerry" setAge : 20];
		OCEqual * equal_c = [[OCEqual alloc] initWithName : @"Jerry" setAge : 20];

		NSLog(@"[equal_a isEqual : equal_b] : %d, [equal_b isEqual : equal_c] : %d",
			[equal_a isEqual : equal_b], [equal_b isEqual : equal_c]);
	}
}

-- 执行结果

localhost:oc_object octopus$ clang -fobjc-arc -framework Foundation OCEqual.m
localhost:oc_object octopus$ ./a.out
2015-10-03 16:58:35.690 a.out[1168:507] int_a == double_a : 1
2015-10-03 16:58:35.693 a.out[1168:507] str_a == str_b : 1, str_c == str_d : 0, [str_c isEqual : str_d] : 1
2015-10-03 16:58:35.693 a.out[1168:507] [equal_a isEqual : equal_b] : 0, [equal_b isEqual : equal_c] : 1

二. 类别 与 扩展

1. Category 类别

(1) 扩展类簇需求

类簇扩展需求 : 开发过程中有时需要扩展类行为;

-- 继承 : 通过继承, 子类在父类基础上添加方法 或者 重写父类方法;

-- 问题 : 如果想要为父类增加一个方法, 子类同时也继承这些方法, 此时使用继承就满足不了这个功能了;

-- 类簇 : OC 中没有接口, 需要接口时, 就会选择定义一个父类, 以该父类派生 N 个子类, 该系列的类被成为 类簇;

-- 类簇扩展方法 : 为父类增加方法, 类簇中得子类同时也增加该方法, 扩展类簇中得父类是最合适的方法;

(2) Category 类别

类别 (category) 简介 :

-- 作用 : 为现有类添加方法, 不需要访问原有类代码, 不需要继承;

-- 有点 : 动态地为现有类添加方法, 将类定义模块化 分布到多个文件中;

(3) Category 类别 接口 语法格式

类别 (category) 接口部分语法格式 :

-- 接口文件类命名 : "类名+类别名.h", 如 要扩展 OCPerson 类, 类别名为 SB, 那么接口文件名就是 "OCPerson+SB.h";

-- 示例 :

@interface 已有类 (类别名)
//方法定义
...
@end

-- 类别名 : 必须是项目中没有的类, 定义类别时使用的类名, 必须是已有的类;

-- 圆括号 : 类别名 定义在 需要扩展的已有类之后, 必须使用圆括号括起来;

-- 定义内容 : 类别中一般情况下只定义方法;

(4) Category 类别 实现类 语法格式

类别 (category) 实现部分语法格式 :

-- 实现类文件命名 : "类名+类别名.m", 如 要扩展 OCPerson 类, 类别名为 SB, 那么接口文件名就是 "OCPerson+SB.m";

-- 示例 :

@implementation 已有类 (类别名)
//方法定义
...
@end

(5) Category 类别 注意点

注意事项 :

-- 影响范围 : 通过 category 添加新方法后, 会影响到 指定的被扩展的类, 同时也会影响到其子类;

-- 多个类别 : 一个类可以 对应多个类别, 这些类别都可以为类增加方法定义;

-- 类别优点 : 进行模块化设计, 调用私有方法, 实现非正式协议;

(6) Category 扩展 NSNumber 示例

NSNumber 扩展示例 : 为其添加一个计算圆面积的方法;

-- NSNumber+SB.h :

/*************************************************************************
    > File Name: NSNumber+SB.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 六 10/ 3 18:58:53 2015
 ************************************************************************/
#import <Foundation/Foundation.h>
@interface NSNumber (SB)
- (NSNumber *) circleAera : (double) radius;
@end

-- NSNumber+SB.m

/*************************************************************************
    > File Name: NSNumber+SB.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 六 10/ 3 19:02:05 2015
 ************************************************************************/
 #import "NSNumber+SB.h"
 @implementation NSNumber (SB)

 - (NSNumber *) circleAera : (double) radius
 {
	double aera = 3.1415926 * radius * radius;
	return [NSNumber numberWithDouble : aera];
 }

 @end

-- NSNumber+SBTest.m

/*************************************************************************
    > File Name: NSNumber+SBTest.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 六 10/ 3 19:08:19 2015
 ************************************************************************/
#import <Foundation/Foundation.h>
#import "NSNumber+SB.h"

int main(int argc, char * argv[])
{
	@autoreleasepool {
		NSNumber * num = [NSNumber numberWithInt : 3];
		NSNumber * circleAera = [num circleAera : 1];
		NSLog(@"%@", circleAera);
	}
}

-- 执行结果

localhost:oc_object octopus$ clang -fobjc-arc -framework Foundation NSNumber+SB.m NSNumber+SBTest.m
localhost:oc_object octopus$ ./a.out
2015-10-03 19:18:13.625 a.out[1333:507] 3.1415926

2. Category 类别实际用法

(1) 类的模块化设计

模块化设计简介 :

-- 实现部分唯一 : 定义一个类是, 使用 "类名.h" 定义接口部分, 使用 "类名.m" 定义实现部分, 不能将实现部分定义在多个 ".m" 后缀 文件中;

-- 文件臃肿 : 如果类很大, 将所有的代码放在一个 "类名.m" 文件中, 非常难维护;

(2) 调用私有方法

私有方法调用简介 :

-- 私有方法 : 接口中没有定义, 在实现部分定义的方法是 私有方法, 不允许被外部调用;

-- 调用私有方法一 : 使用 NSObject 的 "performSelector :"执行调用, 也是可以调用私有方法的, 不过此方法会避开语法检查, 导致未知问题;

(3) 调用私有方法 代码示例

代码示例 :

-- OCPrivate.h :

/*************************************************************************
    > File Name: OCPrivate.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 日 10/ 4 06:55:34 2015
 ************************************************************************/
#import <Foundation/Foundation.h>

@interface OCPrivate : NSObject
@property (nonatomic, copy) NSString * name;
-(void) info;
@end

-- OCPrivate.m :

/*************************************************************************
    > File Name: OCPrivate.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 日 10/ 4 06:57:48 2015
 ************************************************************************/
#import "OCPrivate.h"

@implementation OCPrivate
@synthesize name;

- (void) info
{
	NSLog(@"name : %@", self.name);
}

- (void) speak
{
	NSLog(@"Hello World !");
}
@end

-- OCPrivate+SB.h :

/*************************************************************************
    > File Name: OCPrivate+SB.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 日 10/ 4 07:19:35 2015
 ************************************************************************/
#import "OCPrivate.h"
@interface OCPrivate (SB)
- (void) speak;
@end

-- OCPrivateTest.m :

/*************************************************************************
    > File Name: OCPrivateTest.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 日 10/ 4 07:22:04 2015
 ************************************************************************/
#import "OCPrivate+SB.h"

int main(int argc, char * argv[])
{
	@autoreleasepool
	{
		OCPrivate * priva = [[OCPrivate alloc] init];
		priva.name = @"Tom";
		[priva info];
		[priva speak];

	}
}

3. extension 扩展

(1) extension 简介

extension 简介 :

-- 作用 : 扩展相当于匿名类别;

-- 语法 :

@interface 已有类 ()
{
	//实例变量 ...
}
// 方法定义 ...
@end

-- 用法 : 定义两个头文件, OCExtension.h 和 OCExtension+speak.h, OCExtension.m 导入 OCExtension+speak.h 头文件;

(2) extension 源码示例

源码示例 :

-- OCExtension.h :

/*************************************************************************
    > File Name: OCExtension.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 日 10/ 4 08:18:43 2015
 ************************************************************************/
#import <Foundation/Foundation.h>
@interface OCExtension : NSObject
@property (nonatomic, copy) NSString * name;
@property (nonatomic, assign) int age; 

- (void) info;
@end

-- OCExtension+speak.h :

/*************************************************************************
    > File Name: OCExtension+speak.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 日 10/ 4 08:31:37 2015
 ************************************************************************/
#import "OCExtension.h"
@interface OCExtension ()
@property (nonatomic, copy) NSString * home;
- (void) speak : (NSString *) content;
@end

-- OCExtension.m :

/*************************************************************************
    > File Name: OCExtension.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 日 10/ 4 08:27:50 2015
 ************************************************************************/
#import "OCExtension+speak.h"
@implementation OCExtension
@synthesize name;
@synthesize age;
@synthesize home;
- (void) info
{
	NSLog(@"info : name : %@ , age : %d", self.name, self.age);
}
- (void) speak : (NSString *) content
{
	NSLog(@"%@ speak %@", self.name, content);
}
@end

-- OCExtensionTest.m :

/*************************************************************************
    > File Name: OCExtensionTest.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 日 10/ 4 13:20:42 2015
 ************************************************************************/
#import "OCExtension+speak.h"

int main(int argc, char * argv[])
{
	@autoreleasepool {
		OCExtension * extension = [[OCExtension alloc] init];
		extension.name = @"Tom";
		extension.age = 18;
		extension.home = @"China";

		[extension info];
		[extension speak : @"Are you fucking kidding me"];
	}
}

三. 协议 与 委托

1. 类别实现非正式协议

(1) 非正式协议简介

协议简介 :

-- 作用 : OC 中得协议作用相当于其它语言中得接口;

-- 协议表现 : 协议定义的是 多个类 共同的行为规范, 通常定义一组公用方法, 这些方法都没有实现, 方法由类来实现;

非正式协议简介 :

-- 创建 NSObject 类别 : 以 NSObject 为基础, 为 NSObject 创建类别, 为该类别指定新增方法, 即给所有的 NSObject 子类增加了新方法;

-- 实现 NSObject 类别 : 实现 NSObject 类别时, 实现该列别下地所有方法, 即之前在 NSObject 类别中定义的方法;

(2) 非正式协议代码示例

非正式协议代码示例 :

-- NSObject+speak.h : 为 NSObject 定义的类别接口, 所有的继承 NSObject 的类都必须实现该类别中得抽象方法;

/*************************************************************************
    > File Name: NSObject+speak.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 08:55:48 2015
 ************************************************************************/
#import <Foundation/Foundation.h>
@interface NSObject (speak)
- (void) speak;
@end

-- OCNSObjectProtocal.h : NSObject 子类接口, 该接口继承 NSObject 类, 注意 需要导入 NSObject+speak.h 头文件;

/*************************************************************************
    > File Name: OCNSObjectProtocal.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 09:00:37 2015
 ************************************************************************/
#import "NSObject+speak.h"
@interface OCNSObjectProtocal : NSObject
@end

-- OCNSObjectProtocal.m : OCNSObjectProtocal 实现类, 在该实现类中必须实现 类别中定义的 speak 方法;

/*************************************************************************
    > File Name: OCNSObjectProtocal.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 09:02:03 2015
 ************************************************************************/
#import "OCNSObjectProtocal.h"
@implementation OCNSObjectProtocal
- (void) speak
{
	NSLog(@"Speak Hello World");
}
@end

-- OCNSObjectProtocalTest.m : 测试类, 测试以上代码是否可以执行;

/*************************************************************************
    > File Name: OCNSObjectProtocalTest.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 09:04:31 2015
 ************************************************************************/
#import "OCNSObjectProtocal.h"

int main(int argc, char * argv[])
{
	@autoreleasepool {
		OCNSObjectProtocal * obj = [[OCNSObjectProtocal alloc] init];
		[obj speak];
	}
}

-- 执行结果 :

bogon:oc_object octopus$ clang -fobjc-arc -framework Foundation OCNSObjectProtocal.m OCNSObjectProtocalTest.m
bogon:oc_object octopus$ ./a.out
2015-10-05 09:06:44.895 a.out[2100:507] Speak Hello World

2. 定义正式协议

(1) 正式协议语法

正式协议语法 :

-- 语法 :

@protocol 协议名称 <父类协议1, 父类协议2 ...>
// N 个协议方法
@end

-- 协议名称规范 : 采用与类名相同的命名规则;

-- 继承规则 : 一个协议 可以有 多个父类协议, 协议只能继承协议, 不能继承类;

-- 方法规则 : 协议中只能定义抽象方法, 不能定义方法实现, 既可以定义类方法, 也可以定义实例方法;

(2) 实现协议

实现协议语法 :

-- 语法 :

@interface 类名 : 父类 <协议1, 协议2...>

-- 对应关系 : 一个类可以实现多个协议;

(3) 声明协议变量

变量声明 :

-- 使用原变量声明 : "变量名 * 对象名" , 如 "OCCat * cat";

-- 使用协议定义 : "NSObject <协议1, 协议2 ...> * 对象名", 如 "NSObject<OCProtocolCat> * cat";

-- 使用 id 类型定义 : "id<OCProtocolCat> 对象名", 如 "id<OCProtocolCat> cat", 注意此处没有指针标识;

(4) 正式协议实现代码

代码示例 :

-- OCAnimalProtocol.h : 最基础的协议1;

/*************************************************************************
    > File Name: OCAnimalProtocol.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 09:25:41 2015
 ************************************************************************/
#import <Foundation/Foundation.h>
@protocol OCAnimalProtocol
- (void) name;
- (void) age;
@end

-- OCProtocolBord.h : 最基础的协议2;

/*************************************************************************
    > File Name: OCProtocolBord.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 09:40:08 2015
 ************************************************************************/
#import <Foundation/Foundation.h>
@protocol OCProtocolBord
- (void) fly;
@end

-- OCProtocolCat.h : 该协议实现了 上面的两个协议;

/*************************************************************************
    > File Name: OCProtocolCat.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 09:45:17 2015
 ************************************************************************/

#import <Foundation/Foundation.h>
#import "OCAnimalProtocol.h"
#import "OCProtocolBord.h"

@protocol OCProtocolCat <OCAnimalProtocol, OCProtocolBord>
- (void) purr;
@end

-- OCCat.h : OCCat 类接口部分, 生命了该类 实现协议 OCProtocolCat 协议;

/*************************************************************************
    > File Name: OCCat.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 09:50:21 2015
 ************************************************************************/

#import "OCProtocolCat.h"

@interface OCCat : NSObject <OCProtocolCat>
@end

-- OCCat.m : OCCat 类实现部分;

/*************************************************************************
    > File Name: OCCat.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 09:52:45 2015
 ************************************************************************/

 #import "OCCat.h"

 @implementation OCCat
 - (void) name
 {
	NSLog(@"name : cat");
 }
 - (void) age
 {
	NSLog(@"age : 18");
 }
 - (void) fly
 {
	NSLog(@"cat fly");
 }
 - (void) purr
 {
	NSLog(@"cat purr");
 }
 @end

-- OCProtocolTest.m : 以上类的功能测试类;

/*************************************************************************
    > File Name: OCProtocolTest.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 10:14:38 2015
 ************************************************************************/

#import <Foundation/Foundation.h>
#import "OCCat.h"

int main(int argc, char * argv[])
{
	@autoreleasepool {
		OCCat * cat = [[OCCat alloc] init];
		[cat name];
		[cat age];
		[cat fly];
		[cat purr];

		NSObject<OCProtocolCat> * cat1 = [[OCCat alloc] init];
		[cat1 name];
		[cat1 age];
		[cat1 fly];
		[cat1 purr];

		id<OCAnimalProtocol> cat2 = [[OCCat alloc] init];
		[cat2 name];
		[cat2 age];
	}
}

-- 执行结果 :

bogon:6.4 octopus$ clang -fobjc-arc -framework Foundation OCCat.m OCProtocolTest.m
bogon:6.4 octopus$ ./a.out
2015-10-05 10:24:20.099 a.out[2271:507] name : cat
2015-10-05 10:24:20.101 a.out[2271:507] age : 18
2015-10-05 10:24:20.102 a.out[2271:507] cat fly
2015-10-05 10:24:20.102 a.out[2271:507] cat purr
2015-10-05 10:24:20.102 a.out[2271:507] name : cat
2015-10-05 10:24:20.103 a.out[2271:507] age : 18
2015-10-05 10:24:20.103 a.out[2271:507] cat fly
2015-10-05 10:24:20.104 a.out[2271:507] cat purr
2015-10-05 10:24:20.104 a.out[2271:507] name : cat
2015-10-05 10:24:20.104 a.out[2271:507] age : 18

3. 委托

委托概念 : 定义协议的类定义协议的方法 委托给 实现协议的类;

-- 好处 : 类具有更好地通用性, 具体的动作交给实现类完成;

创建工程 :

-- 欢迎界面, 选择 Create a new xcode project;

【IOS 开发】Objective - C 面向对象高级特性 - 包装类 | 类处理 | 类别 | 扩展 | 协议 | 委托 | 异常处理 | 反射

-- 创建一个 OS 下地 Cocoa Application :

【IOS 开发】Objective - C 面向对象高级特性 - 包装类 | 类处理 | 类别 | 扩展 | 协议 | 委托 | 异常处理 | 反射

-- 创建 工程 :

【IOS 开发】Objective - C 面向对象高级特性 - 包装类 | 类处理 | 类别 | 扩展 | 协议 | 委托 | 异常处理 | 反射

项目中得源文件 :

-- main.m : main() 函数入口;

-- OCAppDelegate.h : OCAppDelegate 类接口文件;

-- OCAppDelegate.m : OCAppDelegate 类实现部分;

【IOS 开发】Objective - C 面向对象高级特性 - 包装类 | 类处理 | 类别 | 扩展 | 协议 | 委托 | 异常处理 | 反射

代码示例 :

-- 前置操作 : 删除 MainMenu.xib 文件, 删除 Hello-Info.plist 中的 MainMenu 选项;

【IOS 开发】Objective - C 面向对象高级特性 - 包装类 | 类处理 | 类别 | 扩展 | 协议 | 委托 | 异常处理 | 反射

-- OCAppDelegate.h :

//
//  OCAppDelegate.h
//  Hello
//
//  Created by octopus on 15-10-5.
//  Copyright (c) 2015年 www.octopus.org.cn. All rights reserved.
//

#import <Cocoa/Cocoa.h>

//该接口 实现 NSApplicationDelegate 协议
@interface OCAppDelegate : NSObject <NSApplicationDelegate>
//定义窗口
@property (strong) NSWindow *window;

@end

-- OCAppDelegate.m :

//
//  OCAppDelegate.m
//  Hello
//
//  Created by octopus on 15-10-5.
//  Copyright (c) 2015年 www.octopus.org.cn. All rights reserved.
//

#import "OCAppDelegate.h"

@implementation OCAppDelegate
@synthesize window;

//应用加载完成时回调的方法
- (void)applicationWillFinishLaunching:(NSNotification *)notification
{
    //设置窗口属性
    self.window = [[NSWindow alloc] initWithContentRect :
                   NSMakeRect(300, 300, 320, 200)
                   styleMask:(NSTitledWindowMask | NSMiniaturizableWindowMask |NSClosableWindowMask)
                   backing:NSBackingStoreBuffered
                   defer:NO];
    self.window.title = @"Hello World";

    //设置文本框属性
    NSTextField * label = [[NSTextField alloc] initWithFrame:NSMakeRect(60, 120, 200, 60)];
    [label setSelectable:YES];
    [label setBezeled:YES];
    [label setDrawsBackground:YES];
    [label setStringValue:@"HELLO WORLD"];

    //设置按钮属性
    NSButton * button = [[NSButton alloc] initWithFrame:NSMakeRect(120, 40, 80, 30)];
    button.title = @"OCTOPUS";
    [button setBezelStyle:NSRoundedBezelStyle];
    [button setBounds:NSMakeRect(120, 40, 80, 30)];

    //将 文本框 和 按钮 添加到窗口中
    [self.window.contentView addSubview:label];
    [self.window.contentView addSubview:button];
}

//加载完成时回调的方法
- (void) applicationDidFinishLaunching:(NSNotification *)notification
{
    //显示窗口
    [self.window makeKeyAndOrderFront:self];
}

@end

-- main.m :

//
//  main.m
//  Hello
//
//  Created by octopus on 15-10-5.
//  Copyright (c) 2015年 www.octopus.org.cn. All rights reserved.
//

#import <Cocoa/Cocoa.h>
#import "OCAppDelegate.h"

int main(int argc, const char * argv[])
{
    @autoreleasepool {
        //创建一个实现了 NSApplicationDelegate 协议的对象
        OCAppDelegate * delegate = [[OCAppDelegate alloc] init];
        //获取 NSApplication 单例对象
        [NSApplication sharedApplication];
        //设置代理, 将处理方法委托给 delegate
        [NSApp setDelegate : delegate];
        //开始运行程序
        return NSApplicationMain(argc, argv);
    }

}

-- 运行结果 :

【IOS 开发】Objective - C 面向对象高级特性 - 包装类 | 类处理 | 类别 | 扩展 | 协议 | 委托 | 异常处理 | 反射

四. 异常处理

1. @try ... @catch ... @finally ... 异常捕捉

(1) Objective-C 异常机制

Objective-C 异常机制 :

-- 作用 : 开发者将引发异常的代码放在 @try 代码块中, 程序出现异常 使用 @catch 代码块进行捕捉;

-- 每个代码块作用 : @try 代码块存放可能出现异常的代码, @catch 代码块 异常处理逻辑, @finally 代码块回收资源;

-- 语法示例 :

@try
{
	// 业务逻辑
}
@catch (异常类型名1 ex)
{
	//异常处理代码
}
@catch (异常类型名2 ex)
{
	//异常处理代码
}
// 可以捕捉 N 个 异常 ...
@finally
{
	//回收资源
}

(2) Objective-C 异常处理过程

异常处理过程 :

-- 生成异常对象 : @try 中出现异常, 系统会生成一个异常对象, 该对象提交到系统中 系统就会抛出异常;

-- 异常处理流程 : 运行环境接收到 异常对象时, 如果存在能处理该异常对象的 @catch 代码块, 就将该异常对象交给 @catch 处理, 该过程就是捕获异常, 如果没有 @catch 代码块处理异常, 程序就会终止;

-- @catch 代码块捕获过程 : 运行环境接收到 异常对象 时, 会依次判断该异常对象类型是否是 @catch 代码块中异常或其子类实例, 如果匹配成功, 被匹配的 @catch 就会处理该异常, 都则就会跟下一个 @catch 代码块对比;

-- @catch 处理异常 : 系统将异常对象传递给 @catch 形参, @catch 通过该形参获取异常对象详细信息;

其它注意点 :

-- @try 与 @catch 对应关系 : 一个 @try 代码块 可以对应 多个 @catch 代码块;

-- {} 省略问题 : 异常捕获的 @try @catch @finally 的花括号不可省略;

NSException 异常类 :

-- 简介 : NSException 是 OC 中所有异常的父类;

-- 位置永远在最后 : @catch 代码块捕获异常时查看 异常对象类型是否是 捕获的异常类型 或者其子类, 一旦放在开头, 后面的异常永远不可能捕获;

(3) 异常信息访问

异常信息访问 :

-- name : 返回异常的详细名称;

-- reason : 返回异常引发的原因;

-- userInfo : 返回异常的用户信息, 一个 NSDictionary 对象;

(4) 使用 finally 回收资源

回收物理资源 : @try 代码块中打开物理资源, 数据库 网络连接 文件等, 都需要回收, 在 @finally 中回收最好;

-- 回收位置分析 : 如果再 @try 中回收, 出现异常, 异常后面的代码无法执行, @catch 中回收, 如果不出现异常, 该代码块就不会执行; 因此 finally 中是必执行的代码, 在这里回收最合适;

(5) 异常代码示例

异常代码示例 :

-- OCAnimal.h : 定义协议;

/*************************************************************************
    > File Name: OCAnimal.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 16:30:02 2015
 ************************************************************************/
#import <Foundation/Foundation.h>

@protocol OCAnimal
@optional
- (void) run;
@end

-- OCCat.h : 定义 OCCat 接口;

/*************************************************************************
    > File Name: OCCat.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 16:33:59 2015
 ************************************************************************/
#import "OCAnimal.h"

@interface OCCat : NSObject <OCAnimal>
@end

-- OCCat.m : 定义 OCCat 实现类;

/*************************************************************************
    > File Name: OCCat.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 16:36:36 2015
 ************************************************************************/
#import "OCCat.h"

@implementation OCCat
@end

-- OCCatTest.m : 测试类;

/*************************************************************************
    > File Name: OCCatTest.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 16:38:01 2015
 ************************************************************************/
#import "OCCat.h"

int main(int argc, char * argv[])
{
	@autoreleasepool {
		OCCat * cat = [[OCCat alloc] init];
		[cat run];
	}
}

-- 执行结果 :

bogon:6.5 octopus$ clang -fobjc-arc -framework Foundation OCCat.m OCCatTest.m
bogon:6.5 octopus$ ./a.out
2015-10-05 16:39:23.589 a.out[2985:507] -[OCCat run]: unrecognized selector sent to instance 0x7fd7a3401870
2015-10-05 16:39:23.611 a.out[2985:507] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[OCCat run]: unrecognized selector sent to instance 0x7fd7a3401870'
*** First throw call stack:
(
	0   CoreFoundation                      0x00007fff903dd25c __exceptionPreprocess + 172
	1   libobjc.A.dylib                     0x00007fff8ecdbe75 objc_exception_throw + 43
	2   CoreFoundation                      0x00007fff903e012d -[NSObject(NSObject) doesNotRecognizeSelector:] + 205
	3   CoreFoundation                      0x00007fff9033b044 ___forwarding___ + 452
	4   CoreFoundation                      0x00007fff9033adf8 _CF_forwarding_prep_0 + 120
	5   a.out                               0x0000000108575efd main + 109
	6   libdyld.dylib                       0x00007fff851f35fd start + 1
)
libc++abi.dylib: terminating with uncaught exception of type NSException
Abort trap: 6

(6) 异常捕获代码示例

异常捕获取示例 : 该示例扔使用上面的 OCAnimal.h, OCCat.h, OCCat.m 示例;

-- OCCatTest.m :

/*************************************************************************
    > File Name: OCCatTest.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 16:38:01 2015
 ************************************************************************/
#import "OCCat.h"

int main(int argc, char * argv[])
{
	@autoreleasepool {
		@try
		{
			OCCat * cat = [[OCCat alloc] init];
			[cat run];
		}
		@catch (NSException * ex)
		{
			NSLog(@"exception name : %@, reason : %@", ex.name, ex.reason);
		}
		@finally
		{
			NSLog(@"finally execute");
		}
		NSLog(@"success");
	}
}

-- 执行结果 :

bogon:6.5 octopus$ clang -fobjc-arc -framework Foundation OCCat.m OCCatTest.m
bogon:6.5 octopus$ ./a.out
2015-10-05 16:53:46.850 a.out[3008:507] -[OCCat run]: unrecognized selector sent to instance 0x7f884bc018b0
2015-10-05 16:53:46.853 a.out[3008:507] exception name : NSInvalidArgumentException, reason : -[OCCat run]: unrecognized selector sent to instance 0x7f884bc018b0
2015-10-05 16:53:46.853 a.out[3008:507] finally execute
2015-10-05 16:53:46.854 a.out[3008:507] success

2. 抛出自定义异常

(1) 自定义异常语法

自定义异常抛出 :

-- 语法 :

@throw 异常对象;

(2) 自定义异常代码示例

自定义异常代码示例 :

-- OCException.h 接口 :

/*************************************************************************
    > File Name: OCException.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 16:58:24 2015
 ************************************************************************/
#import <Foundation/Foundation.h>

@interface OCException : NSException
@end

-- OCException.m 实现类 :

/*************************************************************************
    > File Name: OCException.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 17:03:15 2015
 ************************************************************************/
#import "OCException.h"

@implementation OCException
@end

-- OCCatTest.m 测试类 :

/*************************************************************************
    > File Name: OCCatTest.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 16:38:01 2015
 ************************************************************************/
#import "OCException.h"

int main(int argc, char * argv[])
{
	@autoreleasepool
	{
		@throw [[OCException alloc]
			initWithName : @"OCException"
			reason : @"this reason is imporant"
			userInfo : nil];
	}
}

-- 执行结果 :

bogon:6.5 octopus$ clang -fobjc-arc -framework Foundation OCException.m OCCatTest.m
bogon:6.5 octopus$ ./a.out
2015-10-05 17:04:12.432 a.out[3040:507] *** Terminating app due to uncaught exception 'OCException', reason: 'this reason is imporant'
*** First throw call stack:
(
	0   CoreFoundation                      0x00007fff903dd25c __exceptionPreprocess + 172
	1   libobjc.A.dylib                     0x00007fff8ecdbe75 objc_exception_throw + 43
	2   a.out                               0x00000001062abef7 main + 135
	3   libdyld.dylib                       0x00007fff851f35fd start + 1
)
libc++abi.dylib: terminating with uncaught exception of type OCException
Abort trap: 6

五. Objective-C 反射

1. 获取 Class

(1) 程序 与 环境 交互方式

程序 与 运行环境交互方式 :

-- 通过 OC 源码 : 编写 OC 源码, 编译器编译, 运行在运行环境中;

-- 通过 NSObject 动态编程 : NSObject 是所有类的基类, 所有对象都可以直接调用 NSObject 方法;

-- 调用 运行时函数 动态编程 : 运行时系统是动态库, 可以直接调用这些动态共享库;

(2) 获取 Class 方式

获取 Class 方式 :

-- 通过类名 : 使用 "Class NSClassFromString (NSString * aClassName)" 函数获取 Class 对象, 传入 类名 字符串;

-- class 类方法 : 调用类方法 class, 调用方式 [NSString class];

-- class 对象方法 : 调用对象的 class 方法, 调用方式 [@"hello" class];

-- 推荐使用第二种方式 : 代码更安全, 编译阶段就可以检查 Class 是否存在, 程序性能高;

(3) 获取 Class 代码示例

代码示例 :

/*************************************************************************
    > File Name: OCGetClass.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 23:31:51 2015
 ************************************************************************/

#import <Foundation/Foundation.h>

int main(int argc, char * argv[])
{
	@autoreleasepool {
		//通过 NSClassFromString 方法传入 类名字符串 获取 Class 对象
		Class clazz = NSClassFromString(@"NSDate");
		NSLog(@"%@", clazz);
		id date = [[clazz alloc] init];
		NSLog(@"date : %@", date);

		NSString * str = @"hello";
		// 通过调用 类 或 对象的 class 方法
		NSLog(@"[str class] : %@, [NSString class] : %@", [str class], [NSString class]);
		// 通过调用 类 或 对象的 getter 方法获取, 即用 . 方法获取
		NSLog(@"str.class : %@, NSString.class : %@", str.class, NSString.class);

	}
}

-- 执行结果 :

bogon:6.6 octopus$ clang -fobjc-arc -framework Foundation OCGetClass.m
bogon:6.6 octopus$ ./a.out
2015-10-05 23:39:28.692 a.out[3237:507] NSDate
2015-10-05 23:39:28.699 a.out[3237:507] date : 2015-10-05 15:39:28 +0000
2015-10-05 23:39:28.700 a.out[3237:507] [str class] : __NSCFConstantString, [NSString class] : NSString
2015-10-05 23:39:28.700 a.out[3237:507] str.class : __NSCFConstantString, NSString.class : NSString

2. 检查继承关系

(1) 继承关系判断

继承关系判断方法 :

-- 判断类 : isMemberOfClass 方法, 传入 Class 对象, 判断该对象是否是 Class 对象对应类的实例;

-- 判断类或子类 : isKindOfClass 方法, 传入 Class 对象, 判断该对象是否是 Class 对象对应类 或 子类的实例;

-- 判断协议 : conformsToProtocol 犯法, 传入 Protocol 参数, 传入方法 "@protocol(协议名称)" 或者  "Protocol * NSProtocolFromString(NSString * @"协议名称")" 两种方法获取协议参数;

(2) 继承关系判断代码示例

源码示例 :

-- OCAnimal.h : 定义协议;

/*************************************************************************
    > File Name: OCAnimal.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 23:54:14 2015
 ************************************************************************/
#import <Foundation/Foundation.h>

@protocol OCAnimal
- (void) name;
@end

-- OCCat.h : 定义接口;

/*************************************************************************
    > File Name: OCCat.h
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 23:56:16 2015
 ************************************************************************/
#import "OCAnimal.h"

@interface OCCat : NSObject <OCAnimal>
@end

-- OCCat.m : 定义实现类;

/*************************************************************************
    > File Name: OCCat.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 一 10/ 5 23:58:47 2015
 ************************************************************************/
#import "OCCat.h"

@implementation OCCat
- (void) name
{
	NSLog(@"My name is Tom");
}
@end

-- OCCatMain.m : 测试类;

/*************************************************************************
    > File Name: OCCatMain.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 二 10/ 6 00:00:01 2015
 ************************************************************************/
#import "OCCat.h"

int main(int argc, char * argv[])
{
	@autoreleasepool {
		OCCat * cat = [[OCCat alloc] init];
		NSLog(@"%@", cat.class);

		NSLog(@"cat isMemberOfClass OCCat : %d", [cat isMemberOfClass : OCCat.class]);
		NSLog(@"cat isMemberOfClass NSObject : %d", [cat isMemberOfClass : NSObject.class]);
		NSLog(@"cat isKindOfClass OCCat : %d", [cat isKindOfClass : OCCat.class]);
		NSLog(@"cat isKindOfClass OCCat : %d", [cat isKindOfClass : NSObject.class]);
		NSLog(@"cat conformsToProtocol OCAnimal : %d", [cat conformsToProtocol : @protocol(OCAnimal)]);
	}
}

-- 执行结果 :

bogon:6.6 octopus$ clang -fobjc-arc -framework Foundation OCCat.m OCCatMain.m
bogon:6.6 octopus$ ./a.out
2015-10-06 00:07:56.838 a.out[3337:507] OCCat
2015-10-06 00:07:56.840 a.out[3337:507] cat isMemberOfClass OCCat : 1
2015-10-06 00:07:56.840 a.out[3337:507] cat isMemberOfClass NSObject : 0
2015-10-06 00:07:56.841 a.out[3337:507] cat isKindOfClass OCCat : 1
2015-10-06 00:07:56.841 a.out[3337:507] cat isKindOfClass OCCat : 1
2015-10-06 00:07:56.842 a.out[3337:507] cat conformsToProtocol OCAnimal : 1

3. 动态调用方法

(1) 动态调用成员变量

KVC 机制 : 通过该机制可以动态调用对象的 getter 和 setter 方法, 不论 该变量定义位置 (接口 | 实现) 和 使用何种访问控制符 (private | public), 都可以使用 KVC 访问;

(2) 判断方法是否可调用

判断对象是否可以调用方法 : NSObject 中定义了 respondsToSelector : 方法, 该方法传入 SEL 参数, 该参数代表方法, 如果可以调用 返回 YES, 反之 返回 NO;

获取 SEL 对象方法 :

-- 指令获取 : 使用 @selector 指令获取当前类中指定的方法, 参数是 完整的方法签名关键字, 只有方法名不够;

@selector(setAge:) withObject

-- 方法获取 : 使用 SEL NSSelectorFromString(NSString * aSelectorName) 函数, 根据方法签名关键字字符串获取对应方法;

NSSelectorFromString(@"setAge:")

(3) SEL 动态调用方法

动态调用对象方法 :

-- 动态调用一 : NSObject 的 performSelector : 方法可以调用方法, 需要传入 SEL 对象, 传入参数 可以通过 withObject: 标签传入参数;

[student performSelector : @selector(setAge:) withObject : [NSNumber numberWithInt : 18]];
[student performSelector : NSSelectorFromString(@"setAge:") withObject : [NSNumber numberWithInt : 19]];

-- 动态调用二 : objc_msgSend(receiver, selector, ...) 函数调用方法, 参数一 方法调用者, 参数二 调用的方法, 剩余参数 方法参数;

objc_msgSend (student, @selector(setAge:), [NSNumber numberWithInt : 20]);
objc_msgSend (student, NSSelectorFromString(@"setAge:"), [NSNumber numberWithInt : 21]);

-- 注意 : 使用第二种方法需要导入包 "#import <objc/message.h>", 返回浮点数时调用 objc_msgSend_fpret(), 返回结构体数据时 使用 objc_msgSend_stret() 函数;

(4) IMP 动态调用方法

IMP 动态调用方法 简介 :

-- 获取 IMP 对象 : NSObject 定义了 "- (IMP) methodForSelector : (SEL) aSelector :" 方法, 该方法传入 SEL 参数, 返回 IMP 对象;

-- IMP 作用 : IMP 是 OC 方法函数指针变量, 代表函数入口, 通过 IMP 也可以调用函数;

-- IMP 调用方法语法 : "返回值类型 (*指针变量) (id, SEL, ...)" , 参数一 方法调用者, 参数二 方法 SEL 对象, 后面参数是方法参数;

(5) 动态调用方法 示例代码

示例代码 :

-- OCStudent.m : 接口实现类主函数一体;

/*************************************************************************
    > File Name: OCStudent.m
    > Author: octopus
    > Mail: octopus_truth.163.com
    > Created Time: 二 10/ 6 11:19:49 2015
 ************************************************************************/
#import <Foundation/Foundation.h>
#import <objc/message.h>

@interface OCStudent : NSObject
@end

@implementation OCStudent

- (void) setAge : (NSNumber *) age
{
	int age_num = [age intValue];
	NSLog(@"age is : %d", age_num);
}
@end

int main(int argc, char * argv[])
{
	@autoreleasepool
	{
		OCStudent * student = [[OCStudent alloc] init];

		[student performSelector : @selector(setAge:) withObject : [NSNumber numberWithInt : 18]];
		[student performSelector : NSSelectorFromString(@"setAge:") withObject : [NSNumber numberWithInt : 19]];

		objc_msgSend (student, @selector(setAge:), [NSNumber numberWithInt : 20]);
		objc_msgSend (student, NSSelectorFromString(@"setAge:"), [NSNumber numberWithInt : 21]);
	}
}

-- 执行结果 : 有报警;

bogon:6.6 octopus$ clang -fobjc-arc -framework Foundation OCStudent.m
OCStudent.m:29:12: warning: performSelector may cause a leak because its selector is unknown [-Warc-performSelector-leaks]
                [student performSelector : NSSelectorFromString(@"setAge:") withObject : [NSNumber numberWithInt : 19]];
                         ^
OCStudent.m:29:30: note: used here
                [student performSelector : NSSelectorFromString(@"setAge:") withObject : [NSNumber numberWithInt : 19]];
                                           ^
1 warning generated.
bogon:6.6 octopus$ ./a.out
2015-10-06 12:00:41.669 a.out[747:507] age is : 18
2015-10-06 12:00:41.671 a.out[747:507] age is : 19
2015-10-06 12:00:41.671 a.out[747:507] age is : 20
2015-10-06 12:00:41.672 a.out[747:507] age is : 21