学习目标
1.【掌握】延展Extension
2.【掌握】block类型
3.【掌握】block修改外部变量
4.【理解】block的应用场景
5.【掌握】协议protocol
6.【掌握】@protocol类型限制
7.【理解】代理模式Delegation
一、延展Extension
延展(Extension)是一个特殊的分类,延展没有名字(匿名分类),而且只能有方法的声明,不能有单独的方法实现,与本类共享一个方法实现。我们可以使用@property生成私有属性和对应的getter、setter方法。所以我们常常将延展写在类的实现文件中,用于私有化类的成员(属性和方法),仅限本类访问。
//用于类的实现文件中 #import "Person.h" //定义延展 @interface Person () //延展里声明属性和方法 @end @implementation Person //实现方法 @end
二、block类型
block是OC中新增的一个数据类型,可以声明block变量来存储一段代码,这段代码可以有参数,也可以有返回值,也可以都没有。并且可以通过这个block变量来执行存储在变量中的代码,同时block变量也可以作为一个参数进行传递。在使用时,我们常常会使用typedef来声明一个block类型的别名,这样就可以把符合特定要求的代码块保存在block变量里作为方法的参数进行传递,还能简写代码。
语法:
返回值类型 (^block变量名)(参数列表) = ^代码段返回值类型(参数列表) {
代码段;
};
block类型的简单使用
#import <Foundation/Foundation.h> int main(){ @autoreleasepool { //声明无参无返回值的block变量myBlock1 void (^myBlock1)(); //声明有两个int类型参数int类型返回值的block变量myBlock2 int (^myBlock2)(int num1, int num2); //block变量的参数可以只写参数类型省略参数名 // int (^myBlock2)(int, int); //无参数无返回值的代码段赋值给myBlock1 myBlock1 = ^void() { NSLog(@"代码段内容1"); }; //有两个int类型参数int类型返回值的代码段赋值给myBlock2 myBlock2 = ^int(int num1, int num2) { return num1 +num2; }; int (^myBlock3)(int num1,int num2); //代码段的返回值类型可以省略,但是最终还是要返回符合要求的数据 myBlock3 = ^(int num1, int num2) { return num1 +num2; }; //代码段的返回值类型可以省略,但是最终还是要返回符合要求的数据 void (^myBlock4)() = ^{ NSLog(@"代码段内容4"); }; //使用block变量 myBlock1(); int result1 = myBlock2(1,2); int result2 = myBlock3(1,2); myBlock4(); NSLog(@"result1 = %d",result1); NSLog(@"result2 = %d",result2); } /* 输出 代码段内容1 代码段内容4 result1 = 3 result2 = 3 */ return 0; }
使用typedef将复杂的block定义简化
#import <Foundation/Foundation.h> //只能存储符合要求的代码段 typedef void (^MyBlock1)(); typedef int (^MyBlock2)(int num1, int num2); int main(){ @autoreleasepool { MyBlock1 block2 = ^ { NSLog(@"无参数无返回值"); }; MyBlock2 block1 = ^(int num1, int num2) { return num1 + num2; }; block2(); int result = block1(1,2); NSLog(@"result = %d",result); /* 无参数无返回值 result = 3 */ } return 0; }
三、block访问外部变量
block可以访问外部变量(block外部),但无法改变外部变量的值。如果想要被修改,需要用__block修饰。
#import <Foundation/Foundation.h> //只能存储符合要求的代码段 typedef void (^MyBlock)(); int main(){ @autoreleasepool { //使用__block修饰一个变量 __block int num = 20; MyBlock sumBlock = ^ { NSLog(@"num = %d",++num);//更改被__block修饰的外部变量的值 }; /* num = 21 */ } return 0; }
四、block的应用场景
block作为方法的参数,可以将调用者的代码段传递到函数内部去使用
#import <Foundation/Foundation.h> //只能存储符合要求的代码段 typedef void (^MyBlock)(); //程序员每天发生的事情 void days(void (^myBlock)()); //将每天上班发生的事情写到block void workDay(int day); int main(){ @autoreleasepool { //循环传入天数调用函数 for (int i = 1; i <= 5; i ++) { workDay(i); } } return 0; } void workDay(int day){ MyBlock myBlock; switch (day) { case 1: myBlock = ^{ NSLog(@"看美女......"); }; break; case 2: myBlock = ^{ NSLog(@"看代码......"); }; break; case 3: myBlock = ^{ NSLog(@"写代码......"); }; break; case 4: myBlock = ^{ NSLog(@"测试代码......"); }; break; case 5: myBlock = ^{ NSLog(@"因美女少离职......"); }; break; default: break; } //调用函数传入block days(myBlock); } //程序员每天发生的事情 void days(void (^myBlock)()){ NSLog(@"起床"); NSLog(@"吃饭"); NSLog(@"坐车"); myBlock(); NSLog(@"下班"); NSLog(@"坐车"); NSLog(@"回家"); NSLog(@"吃饭"); NSLog(@"睡觉"); }
block作为方法或函数的返回值,方法执行完毕之后会有一段代码返回给调用者
#import <Foundation/Foundation.h> typedef void (^ReturnBlock)(); //block作为函数返回值,一般开发中很少用到 ReturnBlock test(){ return ^{ NSLog(@"内容"); }; } int main(){ @autoreleasepool { //调用函数 ReturnBlock test1 = test(); } return 0; }
五、协议protocol
在OC中使用@protocol定义一个协议,遵守此协议的类就拥有了这个协议所有的方法声明。协议跟其他面向对象语言的接口有些类似,只是不一定会实现协议里的方法。
//youProtocol.h文件 #import <Foundation/Foundation.h> @protocol youProtocol <NSObject> //被@required修饰,必须实现的方法,不实现编译时也不报错,用于程序员直接的交流 @required - (void)shout; //被@optional修饰,可选实现的方法,不实现编译时也不报错,用于程序员直接的交流 @optional - (void)drive; @end //myProtocol.h文件 #import <Foundation/Foundation.h> @protocol myProtocol <NSObject> @required - (void)sayHi; - (void)song; @optional - (void)paly; - (void)eat; @end //Dog.h文件 #import <Foundation/Foundation.h> #import "myProtocol.h" @interface Dog : NSObject <myProtocol,youProtocol> @end //Dog.m文件 #import "Dog.h" @implementation Dog //继承了两个协议,6个方法只实现了3个必须实现的 - (void)sayHi { NSLog(@"狗汪汪。。"); } - (void)song { NSLog(@"狗唱。。"); } - (void)shout { NSLog(@"狗叫。。"); } @end
总结:
1.协议通过<>进行实现,并且只能在类的声明上遵守协议,一个类可以同时遵守多个协议,中间通过逗号分隔。
2.一个协议可以遵守自另一个协议,例如上面myProtocol就遵守自NSObject(基协议),如果需要遵守多个协议中间使用逗号分隔。
3.如果父类遵守了某个协议,子类继承了父类所有成员,相当于子类也遵守了这个协议,子类不实现也不会警告。
4.协议中不能定义属性,只能声明方法,不要在协议中使用@property定义属性,协议的主要作用是集中声明方法。
5.我们可以通过关键字@required和@optional分别修饰必须实现的方法和可选实现的方法,如果不修饰则默认是@required。
六、@protocol类型限制
要求某个指针保存的是遵守了指定协议的对象,比如id<协议名> obj;,这个obj指针只能指向遵守了“<协议名>”这个协议的对象。之所以要求对象要遵守某个协议,是因为我们可能会在程序中调用协议中的方法,如果没有遵守并实现其对应方法调用时就会报错。一般调用时我们都会使用respondsToSelector:方法来判断遵守了指定协议的对象是否实现了其对应的方法。
//假如有Person类,如果Person没有同时遵守sportProtocol和playProtocol协议就报错 id<sportProtocol,playProtocol> obj = [[Person alloc] init]; //父类<协议名> *指针名; //这里的指针只能指向遵守了指定协议的对象或子类对象,这里是多态。
七、代理模式Delegation
事实上在OC中协议的更多作用是用于约束一个类必须实现某些方法,而从面向对象的角度而言这个类跟其他面向对象语言的接口(interface)并不一定存在某种自然关系,可能是两个完全不同意义上的事物,这种模式我们称之为代理模式(Delegation)。在Cocoa框架中大量采用这种模式实现数据和UI的分离,而且基本上所有的协议都是以Delegate结尾。
现在假设需要设计一个按钮,在按钮中定义按钮的代理,同时使用协议约束这个代理(事件的触发者)必须实现协议中的某些方法,当按钮被点击就检查代理是否实现了这个方法,如果实现了则调用这个方法。
//JFButtonDelegate.h文件 #import <Foundation/Foundation.h> @class JFButton; @protocol JFButtonDelegate <NSObject> @required -(void)onClick:(JFButton *)button; @end //MyListener.h文件 #import <Foundation/Foundation.h> #import "JFButtonDelegate.h" @interface MyListener : NSObject <JFButtonDelegate> @end //MyListener.m文件 #import "MyListener.h" #import "JFButton.h" @implementation MyListener //实现协议中声明的方法 -(void)onClick:(JFButton *)button{ NSLog(@"调用代理的监听方法,被点击的按钮是:%@",button); } @end //JFButton.h文件 #import <Foundation/Foundation.h> #import "JFButtonDelegate.h" @interface JFButton : NSObject //代理属性,同时约定作为代理的对象必须实现JFButtonDelegate协议 //id可以表示任何一个OC对象类型,类型后面的 <协议名> 用于约束作为这个属性的对象必须实现该协议。 @property (nonatomic,strong) id<JFButtonDelegate> delegate; //点击方法 -(void)click; @end //JFButton.m文件 #import "JFButton.h" @implementation JFButton -(void)click{ NSLog(@"调用按钮对象的点击方法"); //使用respondsToSelector方法可以判断self.delegate实例是否实现了onClick:方法 if([self.delegate respondsToSelector:@selector(onClick:)]){ //如果确认onClich:方法已经实现,则作为JFButton的监听 [self.delegate onClick:self]; } } @end //main.m文件 #import <Foundation/Foundation.h> #import "JFButton.h" #import "MyListener.h" int main(int argc, const char * argv[]) { @autoreleasepool { //实例化按钮和监听对象 JFButton *button = [[JFButton alloc]init]; MyListener *listener = [[MyListener alloc]init]; //降遵守了协议的监听对象作为按钮的代理 button.delegate = listener; //按钮调用点击方法 [button click]; /* 调用按钮对象的点击方法 调用代理的监听方法,被点击的按钮是:<JFButton: 0x100114710> */ } return 0; }