iOS分享 - 对象间的通信之delegate、notificationCenter与block
在项目开发中,常常会涉及到对象之间的通信,而为了降低对象间的耦合,会采用delegate、notificationCenter、block三种方式来进行实现,对于他们的使用,也许大家都能熟练掌握,但是对于如何创建,初学者也许只是一知半解,本文不讲长篇大论,仅通过简单的实例来帮助大家学习三者的使用,希望对尚不了解以上三者的朋友能带来一定的帮助。
一、delegate
估计大家最常用的delegate就是UITableViewDelegate了,那么我们如何自己写一个代理呢?我们常常会在控制器中监听某个视图中按钮的点击,就以此为例。假如在DTestView中有一个Button,我们要在DTestViewController中对这个按钮的点击进行监听,并在点击Button时传递一个数组给DTestViewController,那么代码如下:
//1、在DTestView.h文件中声明协议并创建协议方法
@class DTestView;
@protocol DTestViewDelegate <NSObject>
@optional
/**点击按钮并传递数组信息*/
- (void)dTestViewDidClickBtn:(DTestView*)dTestView withArr:(NSArray *)position;
@end
//2、在DTestView.h文件中声明delegate属性@interface DTestView : UIView@property (nonatomic, weak) id <DTestViewDelegate> delegate;
@end
//3、在DTestView.m文件中Button的点击事件里调用delegate的协议方法
- (void)btnClick {
if ([self.delegate respondsToSelector:@selector(dTestViewDidClickBtn:withArr:)]) {
NSArray *arr = @["test", "delegate"];
[self.delegate dTestViewDidClickBtn:self withArr:arr];
}
}
创建代理总共就3步,怎么样,是不是很简单?但是这之中也有几个需要注意的地方:
1、命名规范:类名+Delegate。
2、@optional关键字:遵循协议后,该协议方法可以不实现,协议方法默认为@required修饰,即遵循协议后必须实现该方法,建议不是必须实现的方法都用@optional修饰。
3、ARC环境下,delegate属性用weak修饰。
4、在调用代理方法前,利用respondsToSelector:方法判断代理方法是否存在。
以上就是创建delegate时的注意事项,大家可以思考下为什么要这么做,如有不明白的地方,欢迎提问,如有不对的地方,欢迎指正。
接下来就是delegate的使用了,简单说下,也是3步:
//1、设置代理
- (void)viewDidLoad {
[super viewDidLoad];
dTestView *view = [[dTestView alloc] init];
view.delegate = self;
[self.view addSubview:view];
}
//2、遵循协议
@interface DTestViewController ()<DTestViewDelegate>
//3、实现协议方法
- (void)dTestViewDidClickBtn:(DtestView *)dTestView withArr:(NSArray *)arr {
NSLog(@"%@", arr);
}
好了,这就是一个简单的delegate的创建与使用。
二、notificationCenter
通知中心相对简单,仍以监听按钮点击并传递一个数组为例,在NTestView中发布通知:
//在Button的addTarget点击事件中发布通知
- (void)btnClick {
NSArray *arr = @[@"test", @"notificationCenter"];
[[NSNotificationCenter defaultCenter] postNotificationName:@"点击按钮" object:nil userInfo:arr];
}
完了?完了,就这么简单,接下来在NTestViewController中添加观察者:
//添加观察者
- (void)viewDidLoad {
[super viewDidLoad];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(nTestViewDidClickBtn:) name:@"点击按钮" object:nil];
}
/**接收到通知时调用该方法*/
- (void)nTestViewDidClickBtn:(NSNotification *)note {
NSArray *arr = note.userInfo;
NSLog(@"%@", arr);
}
//移除观察者
- (void)dealloc
{
[[NSNotificationCenter defaultCenter] removeObserver:self];
}
解释一下参数:
name:通知的名称
object:需要传递的对象
userInfo:发布通知是传递的信息(添加观察者时,默认可以接收该参数)
以上,就是notificationCenter的简单使用。注意:在不需要监听时,一定要移除观察者。
三、block(闭包)
block的简单使用,仍以监听按钮点击并传递一个数组为例:
//1、在BtestView.h文件中,我们可以提供这样一个类方法用于初始化:
@interface BTestView : UIView
/**block属性*/
@property(nonatomic,copy) void(^arrBlock)(NSArray *);
/**初始化方法*/
+ (instancetype)bTestViewWithArrBlock:(void(^)(NSArray *arr))arrBlock;
@end
//2、在BTestView.m文件中,实现该方法:
+ (instancetype)bTestViewWithaArrBlock:(void(^)(NSArray *arr))arrBlock {
BTestView *view = [[BTestView alloc] init];
_arrBlock = arrBlock;
return view;
}
//3、在Button的addTarge点击事件中调用arrBlock
- (void)clickBtn {
NSArray *arr = @[@"test", @"notificationCenter"];
_arrBlock(arr);
}
在NTestViewController中使用block:
- (void)viewDidLoad {
[super viewDidLoad];
BTestView *view = [BTestView bTestViewWithArrBlock:^(NSArray *arr){
NSLog(@"%@", arr);
}];
[self.view addSubview:view];
}
不知道大家看到block用法的时候有没有一种熟悉感?
注意:block属性要用copy修饰,同时,在使用的时候要注意不要造成循环引用。
至于这三种通信方式具体要用哪一种?这必须结合实际项目才能做出选择,常见的一种说法是:一对多通信用notificationCenter,一对一通信在方法少的情况用block,方法多的情况用delegate。
注意:以上代码均未通过实际测试,本文仅提供简单的使用思路。
iOS分享 - 对象间的通信之delegate、notificationCenter与block的更多相关文章
- NSNotification,NSNotificationCenter的使用、iOS中五种对象间传值的方式
学习内容 NSNitification与NotificationCenter(通知与通知中心) 通知的使用 [[NSNotificationCenter defaultCenter]addObserv ...
- ios delegate 代理模式 观察者模式 不同视图间的通信
delegate,在ios中比比皆是,NSURLConnection(网络请求有),tableView, connectionView,等系统自带 的常见代理.甚至,自己写代码的时候,随意间敲打出了p ...
- iOS开发多线程篇—线程间的通信
iOS开发多线程篇—线程间的通信 一.简单说明 线程间通信:在1个进程中,线程往往不是孤立存在的,多个线程之间需要经常进行通信 线程间通信的体现 1个线程传递数据给另1个线程 在1个线程中执行完特定任 ...
- iOS边练边学--多线程NSOperation介绍,子类实现多线程的介绍(任务和队列),队列的取消、暂停(挂起)和恢复,操作依赖与线程间的通信
一.NSOperation NSOperation和NSOperationQueue实现多线程的具体步骤 先将需要执行的操作封装到一个NSOperation对象中 然后将NSOperation对象添加 ...
- iOS开发多线程篇 04 —线程间的通信
iOS开发多线程篇—线程间的通信 一.简单说明 线程间通信:在1个进程中,线程往往不是孤立存在的,多个线程之间需要经常进行通信 线程间通信的体现 1个线程传递数据给另1个线程 在1个线程中执行完特定任 ...
- C++对象间通信组件,让C++对象“无障碍交流”
介绍 这是很久之前的一个项目了,最近刚好有些时间,就来总结一下吧! 推荐初步熟悉项目后阅读本文: https://gitee.com/smalldyy/easy-msg-cpp 从何而来 这要从我从事 ...
- 如何才能学到Qt的精髓——信号槽之间的无关性,提供了绝佳的对象间通讯方式,QT的GUI全是自己的一套,并且完全开源,提供了一个绝好机会窥视gui具体实现
姚冬,中老年程序员 叶韵.KY Xu.赵奋强 等人赞同 被邀请了很久了,一直在思考,今天终于下决心开始写回答. 这个问题的确是够大的,Qt的代码规模在整个开源世界里也是名列前茅的,这么大的项目其中的精 ...
- 【转】Java学习:Java中的线程之线程间的通信
hello各位小伙伴 今天我们来搞一下 线程之间的通信 ( • ̀ω•́ )✧ 让线程按照我们的想法来执行 两个线程间的通信 这是我们之前的线程. 执行效果:谁抢到资源,谁运行~ 实现线程交替执行: ...
- js 页面间的通信
看了一下公司原来的代码,原页面ajax post返回一个页面完整的HTML,然后再打开一个新页面并输出ajax返回的所有代码到新页面上,在新页面上以表单提交的形式实现重定向. 任凭我想了半天也没想出来 ...
随机推荐
- C++ Primer 5th 第8章 IO库
IO类对象不允许进行拷贝操作. IO类中定义后一些函数和标志,可以用于访问和操作流的状态. 一旦流发生错误,后续IO操作都是失败的. 读写IO对象会改变IO对象的状态. 每个输出流都管理一个缓冲区. ...
- Spring4.0学习笔记(5) —— 管理bean的生命周期
Spring IOC 容器可以管理Bean的生命周期,Spring允许在Bean生命周期的特定点执行定制的任务 Spring IOC 容器对Bean的生命周期进行管理的过程: 1.通过构造器或工厂方法 ...
- twisted(2)--聊天系统
我们今天要做一个聊天系统,这样可以和我们之前flask api那系列文章结合起来:其次,聊天系统最能代表tcpserver,以后可以套用各种模型,比如我们公司做的物联网,其实就是把聊天系统简化一下. ...
- 超级列表框List Ctrl
LVCFMT_CENTER居中对齐 LONG styles; CListCtrl *str=new CListCtrl; str->Create(LVS_ICON, CRect(,,,), ); ...
- the smallest positive number
2520 is the smallest number that can be divided by each of the numbers from 1 to 10 without any rema ...
- XML在数据传输哪些方面会比JSON有优势,在哪些领域更加适合?
XML 跟 JSON 的圣战,或许会成为自 vim/emacs 圣战,cli/gui 圣战等等圣战以来的又一个圣战,而所有的圣战大抵都不会有结果,因为每方都有各自的道理,谁都不服谁. 在我看来,XML ...
- Android读取Assert文件夹下txt文本并变为String的方法
使用场景,在assert文件夹下 有些文本文件,我们需要通过工具类读取出来,然后放到String字符串中,我们该如何操作呢: 直接上代码: 使用方法: MyActivity.readAssertRes ...
- Java---设计模块(装饰模式)
★ 场景和问题 在不对原有对象类进行修改的基础上,如何给一个或多个已有的类对象提供增强额外的功能? ★ 引例 写一个MyBufferedReader类,使它能够对字符流(如FileReader.Inp ...
- ACM2114_S[I](1^3+2^3+3^3)
#include<iostream> using namespace std; int main() { __int64 n,m,i,j,sum; while(cin>>n) ...
- ACM2039_三角形三边关系
#include <iostream> using namespace std; int main(int argc, char* argv[]) { double a,b,c; int ...