實(shí)在不好意思,題目起的太大了,這里只是對Masnory的實(shí)現(xiàn)仿照寫一個最簡單的demo.把實(shí)現(xiàn)思路簡單闡述一下,共同進(jìn)步
各位朋友:源碼在demo,可以去看看.
在Masonry中,實(shí)現(xiàn)是這樣寫的:
UIView *v = [[UIImageView alloc]init];
v.backgroundColor = [UIColor redColor];
[self.view addSubview:v];
//使用 Masonry 設(shè)置
[v mas_makeConstraints:^(MASConstraintMaker *make) {
make.top.mas_equalTo(self.view).offset(200);
make.bottom.mas_equalTo(self.view).offset(-200);
make.left.mas_equalTo(self.view).offset(100);
make.right.mas_equalTo(self.view).offset(-100);
}];
我們想要實(shí)現(xiàn)的效果是:
- 我們在使用過程中需要 先執(zhí)行A方法,再執(zhí)行B方法,或者先執(zhí)行B方法,再執(zhí)行A方法,連寫在一起,就是 鏈?zhǔn)骄幊?/li>
[[某實(shí)例對象 A方法] B方法];
或者是:
[[某實(shí)例對象 B方法] A方法];
2.在OC中 調(diào)用方法都是使用 中括號'[]', 使用小括號'()'的在OC 中是 block,調(diào)用block時候用,函數(shù)編程
//調(diào)用block
block();
//最后實(shí)現(xiàn)的效果是:
某實(shí)例對象.A方法().B方法();
或者是:
某實(shí)例對象.B方法().A方法();
我們一步一步來:
創(chuàng)建一個Person類,自己實(shí)現(xiàn)
1>我們先來說實(shí)現(xiàn)鏈?zhǔn)骄幊?/h4>
思考: 想要實(shí)現(xiàn)鏈?zhǔn)骄幊?就是需要 某個實(shí)例對象,調(diào)用了某個方法,返回的還是這個實(shí)例對象本身,這樣就能繼續(xù)調(diào)用其他的方法了
在Person.h里:
// 鏈?zhǔn)秸Z法. -->就是連續(xù)調(diào)用, [[person run] eat],這種.或是[[person eat] run]這種.需要返回值是它本身
-(Person *)run1;
-(Person *)eat1;
在Person.m里:
//下面返回的是其本身,這樣就能 使用鏈?zhǔn)秸Z法,返回還是器本身,就能再繼續(xù)調(diào)用其他的
-(Person *)run1 {
NSLog(@"%s",__FUNCTION__);
return self;
}
-(Person *)eat1 {
NSLog(@"%s",__FUNCTION__);
return self;
}
在ViewController.m里
//1 >鏈?zhǔn)骄幊?使用鏈?zhǔn)秸Z法
[[person run1] eat1]; //可以這樣寫
[[person eat1] run1]; //也可以調(diào)換順序,這樣寫
2 >我們現(xiàn)在來實(shí)現(xiàn)函數(shù)編程,實(shí)現(xiàn)person.eat2()
思考: 想要使用函數(shù)編程語法, 就是想辦法使用小括號'()'. 在OC中,只有調(diào)用block的時候,才會使用'()',所以我們要使得run2這個方法的返回值是block.
在Person.h里:
//函數(shù)編程 調(diào)用方法使用() 小括號
-(void(^)())run2;
在Person.m里:
-(void (^)())run2 {
//我們要返回的函數(shù)是一個block,在外面才能調(diào)用 person.run2(),這樣寫
void (^runBlock)() = ^{
NSLog(@"run2");
};
return runBlock;
}
在ViewController.m里
//2 >函數(shù)式編程:
person.run2();
3 > 但是 不能繼續(xù) person.run3().eat3.
思考:1 > 需要使得第一個方法返回值也是Person,就是本事,self,才能在函數(shù)編程的繼承上,進(jìn)行鏈?zhǔn)骄幊?/h6>
2> 實(shí)現(xiàn)函數(shù)編程基礎(chǔ)上的鏈?zhǔn)骄幊?需要方法 run3 的返回值是一個block,實(shí)現(xiàn)函數(shù)編程,在block的內(nèi)部返回一個Person,實(shí)現(xiàn)鏈?zhǔn)骄幊?/h6>
在Person.h里:
// 在使用函數(shù)編程編程的 基礎(chǔ)上使用 鏈?zhǔn)骄幊?需要使得返回值是 本事
-(Person *(^)())run3;
-(Person *(^)())eat3;
在Person.m里:
//3.1 實(shí)現(xiàn)函數(shù)編程基礎(chǔ)上的鏈?zhǔn)骄幊?需要方法 run3 的返回值是一個block,實(shí)現(xiàn)函數(shù)編程,在block的內(nèi)部返回一個Person,實(shí)現(xiàn)鏈?zhǔn)骄幊?-(Person *(^)())run3 {
Person *(^runBlock)() = ^{
NSLog(@"--run3");
return self;
};
return runBlock;
}
// 3.2 我們把3.1 內(nèi)部的實(shí)現(xiàn)再寫成一步
-(Person *(^)())eat3 {
return ^{
NSLog(@"-eat3");
return self;
};
}
在ViewController.m里
person.run3().eat3(); //可以這樣 先跑后吃
person.eat3().run3(); //也可以這樣,先吃后跑
4 > 但是還是不能像Masnory那樣有參數(shù),我們再改造.
在Person.h里:
//Person *(^)(double) --> Person *是block的返回值. (double)是參數(shù)
-(Person *(^)(double))run4;
-(Person *(^)(NSString *))eat4;
在Person.m里:
// 4 我們需要使得block帶參數(shù),這樣才能有跟Masnory那樣 equalTo(self).offset(20);的效果
-(Person *(^)(double))run4 {
//*** '大的返回值'是 這個 run4的返回值. '小的返回值'-->'內(nèi)部的返回值'是run4的返回值那個block的返回值
return ^(double distance){
NSLog(@"run4,%f",distance);
return self;
};
}
-(Person *(^)(NSString *))eat4 {
return ^(NSString * food){
NSLog(@"eat4 %@",food);
return self;
};
}
在ViewController.m里
person.run4(100).eat4(@"面包");
person.eat4(@"大米飯").run4(300);
但是,發(fā)現(xiàn)這樣寫block,實(shí)在是一種痛苦,還是使用typedef來實(shí)現(xiàn)吧:(這就是所謂的先苦后甜吧)
在Person.h里:
@class Person;
typedef Person *(^myRunblock)(double distance); //這個distance,可以省略不寫
typedef Person *(^myEatblock)(NSString *);
....
//5, 使用起別名的block,進(jìn)行函數(shù)編程和鏈?zhǔn)骄幊?-(myRunblock)run5;
-(myEatblock)eat5;
在Person.m里:
// 使用起別名的block 再進(jìn)行 函數(shù)編程和鏈?zhǔn)骄幊?比較方便
-(myRunblock)run5 {
return ^(double distance){
NSLog(@"run5 %f",distance);
return self;
};
}
-(myEatblock)eat5 {
return ^(NSString *food){
NSLog(@"eat5 %@",food);
return self;
};
}
在ViewController.m里
//5 但是我們都是在person手寫block的,這樣早控制器中使用的時候函數(shù)編程的時候,沒有提示,我們使用 tydefe個block起別名試試,再說
person.run5(20).eat5(@"饅頭");
person.eat5(@"西瓜").run5(10);
各位朋友:源碼在demo
// 鏈?zhǔn)秸Z法. -->就是連續(xù)調(diào)用, [[person run] eat],這種.或是[[person eat] run]這種.需要返回值是它本身
-(Person *)run1;
-(Person *)eat1;
//下面返回的是其本身,這樣就能 使用鏈?zhǔn)秸Z法,返回還是器本身,就能再繼續(xù)調(diào)用其他的
-(Person *)run1 {
NSLog(@"%s",__FUNCTION__);
return self;
}
-(Person *)eat1 {
NSLog(@"%s",__FUNCTION__);
return self;
}
//1 >鏈?zhǔn)骄幊?使用鏈?zhǔn)秸Z法
[[person run1] eat1]; //可以這樣寫
[[person eat1] run1]; //也可以調(diào)換順序,這樣寫
//函數(shù)編程 調(diào)用方法使用() 小括號
-(void(^)())run2;
-(void (^)())run2 {
//我們要返回的函數(shù)是一個block,在外面才能調(diào)用 person.run2(),這樣寫
void (^runBlock)() = ^{
NSLog(@"run2");
};
return runBlock;
}
//2 >函數(shù)式編程:
person.run2();
2> 實(shí)現(xiàn)函數(shù)編程基礎(chǔ)上的鏈?zhǔn)骄幊?需要方法 run3 的返回值是一個block,實(shí)現(xiàn)函數(shù)編程,在block的內(nèi)部返回一個Person,實(shí)現(xiàn)鏈?zhǔn)骄幊?/h6>
在Person.h里:
// 在使用函數(shù)編程編程的 基礎(chǔ)上使用 鏈?zhǔn)骄幊?需要使得返回值是 本事
-(Person *(^)())run3;
-(Person *(^)())eat3;
在Person.m里:
//3.1 實(shí)現(xiàn)函數(shù)編程基礎(chǔ)上的鏈?zhǔn)骄幊?需要方法 run3 的返回值是一個block,實(shí)現(xiàn)函數(shù)編程,在block的內(nèi)部返回一個Person,實(shí)現(xiàn)鏈?zhǔn)骄幊?-(Person *(^)())run3 {
Person *(^runBlock)() = ^{
NSLog(@"--run3");
return self;
};
return runBlock;
}
// 3.2 我們把3.1 內(nèi)部的實(shí)現(xiàn)再寫成一步
-(Person *(^)())eat3 {
return ^{
NSLog(@"-eat3");
return self;
};
}
在ViewController.m里
person.run3().eat3(); //可以這樣 先跑后吃
person.eat3().run3(); //也可以這樣,先吃后跑
4 > 但是還是不能像Masnory那樣有參數(shù),我們再改造.
在Person.h里:
//Person *(^)(double) --> Person *是block的返回值. (double)是參數(shù)
-(Person *(^)(double))run4;
-(Person *(^)(NSString *))eat4;
在Person.m里:
// 4 我們需要使得block帶參數(shù),這樣才能有跟Masnory那樣 equalTo(self).offset(20);的效果
-(Person *(^)(double))run4 {
//*** '大的返回值'是 這個 run4的返回值. '小的返回值'-->'內(nèi)部的返回值'是run4的返回值那個block的返回值
return ^(double distance){
NSLog(@"run4,%f",distance);
return self;
};
}
-(Person *(^)(NSString *))eat4 {
return ^(NSString * food){
NSLog(@"eat4 %@",food);
return self;
};
}
在ViewController.m里
person.run4(100).eat4(@"面包");
person.eat4(@"大米飯").run4(300);
但是,發(fā)現(xiàn)這樣寫block,實(shí)在是一種痛苦,還是使用typedef來實(shí)現(xiàn)吧:(這就是所謂的先苦后甜吧)
在Person.h里:
@class Person;
typedef Person *(^myRunblock)(double distance); //這個distance,可以省略不寫
typedef Person *(^myEatblock)(NSString *);
....
//5, 使用起別名的block,進(jìn)行函數(shù)編程和鏈?zhǔn)骄幊?-(myRunblock)run5;
-(myEatblock)eat5;
在Person.m里:
// 使用起別名的block 再進(jìn)行 函數(shù)編程和鏈?zhǔn)骄幊?比較方便
-(myRunblock)run5 {
return ^(double distance){
NSLog(@"run5 %f",distance);
return self;
};
}
-(myEatblock)eat5 {
return ^(NSString *food){
NSLog(@"eat5 %@",food);
return self;
};
}
在ViewController.m里
//5 但是我們都是在person手寫block的,這樣早控制器中使用的時候函數(shù)編程的時候,沒有提示,我們使用 tydefe個block起別名試試,再說
person.run5(20).eat5(@"饅頭");
person.eat5(@"西瓜").run5(10);
各位朋友:源碼在demo
// 在使用函數(shù)編程編程的 基礎(chǔ)上使用 鏈?zhǔn)骄幊?需要使得返回值是 本事
-(Person *(^)())run3;
-(Person *(^)())eat3;
//3.1 實(shí)現(xiàn)函數(shù)編程基礎(chǔ)上的鏈?zhǔn)骄幊?需要方法 run3 的返回值是一個block,實(shí)現(xiàn)函數(shù)編程,在block的內(nèi)部返回一個Person,實(shí)現(xiàn)鏈?zhǔn)骄幊?-(Person *(^)())run3 {
Person *(^runBlock)() = ^{
NSLog(@"--run3");
return self;
};
return runBlock;
}
// 3.2 我們把3.1 內(nèi)部的實(shí)現(xiàn)再寫成一步
-(Person *(^)())eat3 {
return ^{
NSLog(@"-eat3");
return self;
};
}
person.run3().eat3(); //可以這樣 先跑后吃
person.eat3().run3(); //也可以這樣,先吃后跑
//Person *(^)(double) --> Person *是block的返回值. (double)是參數(shù)
-(Person *(^)(double))run4;
-(Person *(^)(NSString *))eat4;
// 4 我們需要使得block帶參數(shù),這樣才能有跟Masnory那樣 equalTo(self).offset(20);的效果
-(Person *(^)(double))run4 {
//*** '大的返回值'是 這個 run4的返回值. '小的返回值'-->'內(nèi)部的返回值'是run4的返回值那個block的返回值
return ^(double distance){
NSLog(@"run4,%f",distance);
return self;
};
}
-(Person *(^)(NSString *))eat4 {
return ^(NSString * food){
NSLog(@"eat4 %@",food);
return self;
};
}
person.run4(100).eat4(@"面包");
person.eat4(@"大米飯").run4(300);
@class Person;
typedef Person *(^myRunblock)(double distance); //這個distance,可以省略不寫
typedef Person *(^myEatblock)(NSString *);
....
//5, 使用起別名的block,進(jìn)行函數(shù)編程和鏈?zhǔn)骄幊?-(myRunblock)run5;
-(myEatblock)eat5;
// 使用起別名的block 再進(jìn)行 函數(shù)編程和鏈?zhǔn)骄幊?比較方便
-(myRunblock)run5 {
return ^(double distance){
NSLog(@"run5 %f",distance);
return self;
};
}
-(myEatblock)eat5 {
return ^(NSString *food){
NSLog(@"eat5 %@",food);
return self;
};
}
//5 但是我們都是在person手寫block的,這樣早控制器中使用的時候函數(shù)編程的時候,沒有提示,我們使用 tydefe個block起別名試試,再說
person.run5(20).eat5(@"饅頭");
person.eat5(@"西瓜").run5(10);