关于block,除了网络回调类,对于其他的个人用的都很少,所以很多并不是很清楚了,目前新项目接触到大量的block,所以,看了不少别人写的,希望稍微整理下记录下来。
block代码块主要用于对象之间的通信(反向传值和方法传递)。
相关概念
操作系统中的堆和栈
栈区(stack):由系统自动分配,一般存放函数参数值、局部变量的值等。由编译器自动创建与释放。其操作方式类似于数据结构中的栈,即后进先出、先进后出的原则。(脑补枪械类弹夹更容易理解)
堆区(heap):一般由程序员申请并指明大小,最终也由程序员释放。如果程序员不释放,程序结束时可能会由OS回收。对于堆区的管理是采用链表式管理的,操作系统有一个记录空闲内存地址的链表,当接收到程序分配内存的申请时,操作系统就会遍历该链表,遍历到一个记录的内存地址大于申请内存的链表节点,并将该节点从该链表中删除,然后将该节点记录的内存地址分配给程序。内存管理方面
我们称之为代码块,类似于一个方法,而每一个方法都是在被调用的时候从硬盘到内存,然后去执行,执行完消失,所以方法的内存不用我们管理,也就是说方法是在内存的栈区。所以,block不像OC中的类对象(在堆区),它也是在栈区的。如果我们使用block作为一个对象的属性,我们会使用关键字copy修饰它,因为他在栈区,没办法控制他的消亡,当我们用copy修饰的时候,系统会把该block的实现拷贝一份到堆区,这样我们对应的属性,就拥有的该block的所有权,就可以保证block代码块不会提前消亡。block在传递值的时候
对于基本数据类型的外部变量来说,系统默认传递的仅仅是值,也就是说这个局部变量是不能修改的,如果想修改值,会使用__block来修饰这个变量。这样一来,系统在传递的时候,传的就是外部变量的地址,这样我们就可以修改值了。
对于对象类型,传递的是地址,同时默认对该对象进行了一次强引用。系统进行了强引用,而他又对该对象的内存管理袖手旁观,也就是说,他只做了强引用,但是没有做释放操作。这个时候就会造成内存泄漏。所以,我们在使用对象的时候,在MRC下,都会使用__block修饰,在ARC下,使用__weak修饰,这样一来,系统在传递的时候就不会对该对象进行强引用,避免了内存泄漏。
对象之间的通信,我们一般有三种方式:代理、block和通知。顾名思义,通信即两个对象之间,你让我干什么,我让你干什么。
block和代理的使用是一样的,只不过block相对简单,不用制定协议、写代理方法,同时效率更高。而通知,更灵活,发一个通知,谁都能注册接收通知,然后做事情。
- 什么时候使用代理、什么时候使用block、什么时候使用通知?
在使用代理和block的时候,要意识到一个事情,就是通信的两个对象之间,一定是有关系的(A包含B,或者B包含A), 不然怎么设置代理,怎么实现别的对象的block。所以,当两个需要通信的对象之间有包含关系的时候,考虑代理和block。如果B想让A干不止一件事情,就用代理,如果就是一件事情,没必要制定协议和写代理方法,此时可以考虑block。
有的时候,需要通信的两个对象之间没有关系,或者是一个对象要跟多个对象通信的时候,就要用到通知了。
A控制器,push到B控制器,改变A控制器视图背景颜色:
A:
//
// ViewController.h
// CaiBlockTest
//
// Created by mac on 17/1/5.
// Copyright © 2017年 cai. All rights reserved.
//
#import <UIKit/UIKit.h>
#import "BaseViewController.h"
@interface ViewController : BaseViewController
@end
//
// ViewController.m
// CaiBlockTest
//
// Created by mac on 17/1/5.
// Copyright © 2017年 cai. All rights reserved.
//
#import "ViewController.h"
#import "SecondViewController.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
self.view.backgroundColor = [UIColor orangeColor];
self.title = @"ViewController";
}
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
SecondViewController *secVC = [[SecondViewController alloc] init];
__weak typeof(self) weakSelf = self;
secVC.changeColorBlock = ^(NSString *s, UIColor *color) {
//3.实现
NSLog(@"--%@", s);
weakSelf.view.backgroundColor = color;
};
[self.navigationController pushViewController:secVC animated:YES];
}
- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
@end
B:
//
// SecondViewController.h
// CaiBlockTest
//
// Created by mac on 17/1/5.
// Copyright © 2017年 cai. All rights reserved.
//
#import <UIKit/UIKit.h>
#import "BaseViewController.h"
//定义一个block 1.声明
typedef void(^ChangeVCViewColor)(NSString *s, UIColor *color);
@interface SecondViewController : BaseViewController
@property (nonatomic, copy) ChangeVCViewColor changeColorBlock;
@end
//
// SecondViewController.m
// CaiBlockTest
//
// Created by mac on 17/1/5.
// Copyright © 2017年 cai. All rights reserved.
//
#import "SecondViewController.h"
@interface SecondViewController ()
@end
@implementation SecondViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view.
self.view.backgroundColor = [UIColor redColor];
self.title = @"SecondViewController";
//2.调用
self.changeColorBlock(@"changeColor", [UIColor greenColor]);
}
- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
@end
BaseViewController:
//
// BaseViewController.h
// CaiBlockTest
//
// Created by mac on 17/1/8.
// Copyright © 2017年 cai. All rights reserved.
//
#import <UIKit/UIKit.h>
#import "Define.h"
@interface BaseViewController : UIViewController
@end
//
// BaseViewController.m
// CaiBlockTest
//
// Created by mac on 17/1/8.
// Copyright © 2017年 cai. All rights reserved.
//
#import "BaseViewController.h"
@interface BaseViewController ()
@end
@implementation BaseViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view.
if (self.navigationController.viewControllers.count > 1) {
//非最外层控制器加返回按钮
[self leftItemWithTitle];
//右滑返回
self.navigationController.interactivePopGestureRecognizer.delegate = (id)self;
}
}
#pragma mark -
- (void)leftItemWithTitle
{
UIButton *backBtn = [UIButton buttonWithType:UIButtonTypeCustom];
backBtn.frame = CGRectMake(0, 0, 40, 44);
[backBtn setTitleColor:[UIColor blueColor] forState:UIControlStateNormal];
[backBtn setTitleColor:[UIColor purpleColor] forState:UIControlStateHighlighted];
[backBtn setTitle:@"返回" forState:UIControlStateNormal];
[backBtn addTarget:self action:@selector(actionMethod) forControlEvents:UIControlEventTouchUpInside];
UIBarButtonItem *leftItem = [[UIBarButtonItem alloc] initWithCustomView:backBtn];
self.navigationItem.leftBarButtonItem = leftItem;
}
- (void)actionMethod
{
[self.navigationController popViewControllerAnimated:YES];
}
- (void)dealloc
{
//控制器释放时 打印
NSLog(@"--dealloc: %@", NSStringFromClass([self class]));
}
- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
@end
Define.h:
//
// Define.h
// CaiBlockTest
//
// Created by mac on 17/1/8.
// Copyright © 2017年 cai. All rights reserved.
//
#ifndef Define_h
#define Define_h
/* 显示打印行 */
#if DEBUG
#define NSLog(FORMAT, ...) fprintf(stderr, "[%s:%d行] %s\n",[[[NSString stringWithUTF8String:__FILE__] lastPathComponent] UTF8String], __LINE__, [[NSString stringWithFormat:FORMAT, ##__VA_ARGS__] UTF8String]);
#else
#define NSLog(FORMAT, ...) nil
#endif
#endif /* Define_h */
另一个Demo:
//
// ViewController.m
// CaiBlockStudy
//
// Created by mac on 17/1/5.
// Copyright © 2017年 cai. All rights reserved.
//
#import "ViewController.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
//block的定义
//声明和实现写在一起 类似变量的声明和实现 int a = 10;
//1.定义了一个名字为BlockTest1的block对象
//2.BlockTest1的返回值为int类型
//3.等式右边就是block的具体体现
//4.^为block声明和实现的关键字
//也可以定义其他形式的block,如无返回值或无形参等等
int (^BlockTest1)(int, int) = ^(int num1, int num2) {
return num1 *num2;
};
//声明和实现分开写 就像变量先声明再实现 int a; a = 10;
int (^BlockTest2)(int, int);
BlockTest2 = ^(int num1, int num2) {
return num1 * num2;
};
//调用
int test1 = BlockTest1(2, 9);
int test2 = BlockTest2(100, 2);
NSLog(@"打印结果:%d, %d", test1, test2);
}
- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
@end
--打印结果:
2017-01-08 13:33:10.237 CaiBlockStudy[39557:1569817] 打印结果:18, 200