一. 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;
}
@endint 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;
}
@endint 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;

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

-- 创建 工程 :

项目中得源文件 :

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

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

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

代码示例 :

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

-- 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:NSBackingStoreBuffereddefer: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);}}

-- 运行结果 :

四. 异常处理

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 + 1721   libobjc.A.dylib                     0x00007fff8ecdbe75 objc_exception_throw + 432   CoreFoundation                      0x00007fff903e012d -[NSObject(NSObject) doesNotRecognizeSelector:] + 2053   CoreFoundation                      0x00007fff9033b044 ___forwarding___ + 4524   CoreFoundation                      0x00007fff9033adf8 _CF_forwarding_prep_0 + 1205   a.out                               0x0000000108575efd main + 1096   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 + 1721   libobjc.A.dylib                     0x00007fff8ecdbe75 objc_exception_throw + 432   a.out                               0x00000001062abef7 main + 1353   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);
}
@endint 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

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

  1. Java第四次作业——面向对象高级特性(继承和多态)

    Java第四次作业--面向对象高级特性(继承和多态) (一)学习总结 1.学习使用思维导图对Java面向对象编程的知识点(封装.继承和多态)进行总结. 2.阅读下面程序,分析是否能编译通过?如果不能, ...

  2. Java第四次作业,面向对象高级特性(继承和多态)

    Java第四次作业-面向对象高级特性(继承和多态) (一)学习总结 1.学习使用思维导图对Java面向对象编程的知识点(封装.继承和多态)进行总结. 2.阅读下面程序,分析是否能编译通过?如果不能,说 ...

  3. python面向对象三大特性、类的约束、print带颜色输出及super补充

    面向对象三大特性.类的约束.print带颜色输出及super补充 简述: python面向对象的三大特性: 1.继承是一种创建新类的方式,在python中,新建的类可以继承一个或多个父类,父类又可称为 ...

  4. 【iOS 开发】Objective-C 运算符

    博客地址 : http://blog.csdn.net/shulianghan/article/details/41624613 参考文章 : 1.[iOS 开发]Object-C 运算符 2.[iO ...

  5. 五. 面向对象高级特性4. 接口的概念和使用

    在抽象类中,可以包含一个或多个抽象方法:但在接口(interface)中,所有的方法必须都是抽象的,不能有方法体,它比抽象类更加"抽象". 接口使用 interface 关键字来声 ...

  6. Java第五次作业--面向对象高级特性(抽象类和接口)

    一.学习要点 认真看书并查阅相关资料,掌握以下内容: 掌握抽象类的设计 掌握接口的设计 理解简单工厂设计模式 理解抽象类和接口的区别 掌握包装类的应用 掌握对象的比较方法和比较器的使用 学习使用日期操 ...

  7. JS面向对象高级特性

    对象的创建可以通过两种方式,第一种通过对象初始化的方法: var person={name:"xingoo",age:26,say:function(){console.log(& ...

  8. python魔术方法abstract_python学习之面向对象高级特性和魔术方法

    01_property商品应用.py 分页显示是一种非常常见的浏览和显示大量数据的方法,属于web编程中最常处理的事件之一. 类属性应用需求: 对于京东商城中显示电脑主机的列表页面,每次请求不可能把数 ...

  9. java面向对象高级特性 ---接口案例(学校中有教师和学生两类人,而在职研究生既是教师又是学生)

    学校中有教师和学生两类人,而在职研究生既是教师又是学生.     1)设计两个接口studentinterface和teacherinterface.     2)studentinterface接口 ...

最新文章

  1. 浅谈WPF的VisualBrush
  2. HSSFWorkbook 与 XSSFWorkbook
  3. 尚硅谷Java视频教程导航(学习路线图)
  4. UE4 HTC VIVE 多人联机
  5. SNMP实现交换机的信息采集——MIB
  6. vs code 前端如何以服务器模式打开 [安装服务器] server insteall
  7. HDU2030 汉字统计【文本处理】
  8. [数位dp][状压dp] Jzoj P3458 密码
  9. python 成语库_python实现成语找一找
  10. java c s聊天程序_java----------C/S编程-----简单聊天程序
  11. EC2(elastic compute cloud,弹性计算云,又称EC2实例)
  12. 2020届恒生电子秋季招聘试卷
  13. 量子计算(十五):半导体量子芯片
  14. (五十三)Credit VaR的计算
  15. opencore 启动总是在win_黑苹果OpenCore引导总结
  16. glog logging library for C++
  17. MySQL优化之——函数
  18. Begin UIQ 3.0
  19. MVG(second)读书笔记-3D摄影几何和变换
  20. kali局域网APR攻击三https降级为http+网站账号密码获取

热门文章

  1. 黑马lavarel教程---12、lavarel验证码
  2. Python进阶之一
  3. .NET面试题(三)
  4. [FFmpeg] ffmpeg参数详解
  5. 如何将PHP对象数组转换成普通数组
  6. linux memcached 的安装
  7. MySQL---表操作
  8. wpf中xaml的类型转换器与标记扩展
  9. 哈希表查找速度为什么那么快?快在哪里了?
  10. java_多线程_基于接口的多线程