iOSiOS开发资料收集ios 知识点

传值Delegate/NSNotification/Block/

2017-02-21  本文已影响17人  雷鸣1010

demo下载地址

11.gif

实现了以下iOS页面间传值:
1.委托delegate方式;2.通知notification方式;3.block方式;4.UserDefault或者文件方式;5.单例模式方式;6.通过设置属性,实现页面间传值; 7.KVO通过监听实现传值

在iOS开发中,我们经常会遇到页面间跳转传值的问题,现归纳总结一下:

<一>最简单的设置属性,通过属性传值

情况:A页面跳转到B页面,把A中的值传递给B的某一个属性
方法:在B页面的控制器中,编写对应的属性,在A页面跳转到B页面的地方,给B的属性赋值即可

//SecondViewController.h
@property(nonatomic) NSInteger flag;//当前系统标示(0:其他传值方式;1:block传值方式

在A页面的试图控制器中

//RootViewController.m

- (IBAction)showSecondView:(id)sender {
    SecondViewController *second = [[SecondViewController alloc] initWithNibName:@"SecondViewController" bundle:nil];
    second.delegate = self;
    second.flag = 0;
    [self presentViewController:second animated:YES completion:nil];
}

<二>通过委托delegate的方式实现

情况:A页面跳转到B页面,B页面再跳转回A页面

20130921004015906.png
注意:这里想将B中的值传递给A,就是说在调用代理方法时将B中的值作为代理方法的参数传递给了A;这么说也就是B是委托方,A是代理方

(1)在A页面中设置代理属性,协议和代理方法

@protocol secondViewDelegate
-(void)showName:(NSString *)nameString;
@end


@interface SecondViewController : UIViewController
@property (nonatomic, weak)id<secondViewDelegate> delegate;
@end

(2)选择合适的时机调用代理方法

- (IBAction)delegateMethod:(id)sender {
    if ([self notEmpty]) {
        [self.delegate showName:self.nameTextField.text];
        [self dismissViewControllerAnimated:YES completion:nil];
    }else{
        [self showAlert];
    }
}

(3)在代理方A中实现代理方法

// 实现代理方法
-(void)showName:(NSString *)nameString{
    self.nameLabel.text = nameString;
}

最重要也是最容易忽略的,就是一定要设置delegate的指向。

<三>通过通知notification的方式实现

20130921005718187.png

情况:从B界面往A界面传值

(1)在B页面的控制器中,发送通知:

//SecondViewController.m
- (IBAction)notificationMethod:(id)sender {
    if ([self notEmpty]) {
        [[NSNotificationCenter defaultCenter] postNotificationName:@"ChangeNameNotification" object:self userInfo:@{@"name":self.nameTextField.text}];
        [self dismissViewControllerAnimated:YES completion:nil];
    }else{
        [self showAlert];
    }
}

(2)在A页面的控制器中,注册通知,实现方法:

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view from its nib.
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(ChangeNameNotification:) name:@"ChangeNameNotification" object:nil];
}


-(void)ChangeNameNotification:(NSNotification*)notification{
    NSDictionary *nameDictionary = [notification userInfo];
    self.nameLabel.text = [nameDictionary objectForKey:@"name"];
}

(3)当我们不使用时,要记得删掉通知:

//RootViewController.m
-(void)dealloc{
    [[NSNotificationCenter defaultCenter] removeObserver:self];
}

<四>通过block回调方式实现

情况:假如从A界面往B界面传值,其实就是在A界面执行block的时候将值通过参数的方式传递
此处标注一点,利用block传值时,需要具备以下几个步骤

(1)在A界面定义一个包含参数的block类型,在A界面有一个以block作为参数的方法

typedef void (^ablock)(NSString *str);
@property (nonatomic, copy) ablock block;

(2)在B界面调用A界面这个方法,并且在block里面写上代码,代码里面一定要重点写好如何利用block的参数(就像这里的newColor),完成传值功能

- (IBAction)showSecondWithBlock:(id)sender {
    SecondViewController *second = [[SecondViewController alloc] initWithNibName:@"SecondViewController" bundle:nil];
    [self presentViewController:second animated:YES completion:nil];
    second.block = ^(NSString *str){
        self.nameLabel.text = str;
    };
}

(3)在A界面执行block代码块,将A界面需要传递的值当做block的参数传入

- (IBAction)blockMethod:(id)sender {
    if ([self notEmpty]) {
        if (self.block) {
            self.block(self.nameTextField.text);
            [self dismissViewControllerAnimated:YES completion:nil];
        }
    }else{
        [self showAlert];
    }
}

<五>通过kvo通过监听实现跨界面传值

情况:将B界面的值通过监听的方式传递到A界面
(1):在B界面定义一个全局变量userName,将B界面nameTextField.text的值赋值给useName;

// 我们将赋值过程放在Cancel方法中
- (IBAction)Cancel:(id)sender {
    
    // 触发监听userName的监听机制
    self.userName = self.nameTextField.text;
    
    [self dismissViewControllerAnimated:YES completion:nil];
}

(2)在A界面中实现注册监听对象的方法

// 我们将注册监听对象的方法写在addSecondObserve的方法中,这里是对self.second这个对象的userName属性进行监听
- (IBAction)addSecondObserve:(id)sender {
    
    self.second = [[SecondViewController alloc] initWithNibName:@"SecondViewController" bundle:nil];
    // 注册KVO监听self.second的userName属性的变化
    [self.second addObserver:self forKeyPath:@"userName" options:NSKeyValueObservingOptionNew context:nil];
    
    [self presentViewController:self.second animated:YES completion:nil];
    
    
}

(3)实现当监听的对象即self.second这个对象的userName的值发生改变时调用的方法

// KVO实现监听
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
    //此处监听key对应值的变化情况
    if ([keyPath isEqualToString:@"userName"]) {
        self.nameLabel.text = self.second.userName;
    }
}

(4)清理观察者

//清理观察
- (void)dealloc
{    
    // 清楚KVO中second的观察者
    [self.second removeObserver:self forKeyPath:@"userName"];
}

<六>通过UserDefault实现跨界面传值

(1)通过文件或者UserDefault方式存值(感觉不太适合此类传值,如果要用文件或者UserDefault方式存值的话,可以考虑此方式)

//通过文件或者UserDefault方式存值(感觉不太适合此类传值,如果要用文件或者UserDefault方式存值的话,可以考虑此方式)
- (IBAction)userDefaultMethod:(id)sender {
    if ([self notEmpty]) {
        [[NSUserDefaults standardUserDefaults] setObject:self.nameTextField.text forKey:@"myNameText"];
        [self dismissViewControllerAnimated:YES completion:nil];
    }else{
        [self showAlert];
    }
}

(2) 通过文件或者UserDefault方式取值


-(void)viewDidAppear:(BOOL)animated{
    [super viewDidAppear:animated];
    //如果想测试通过UserDefault方式传值或者通过单例方式传值,取消以下注释即可

    if ([[[NSUserDefaults standardUserDefaults] objectForKey:@"myNameText"] length] != 0) {
        self.nameLabel.text = [[NSUserDefaults standardUserDefaults] objectForKey:@"myNameText"];
        [[NSUserDefaults standardUserDefaults] setObject:@"" forKey:@"myNameText"];
    }
    }

<七>通过单例模式方式实现跨界面传值

(1)实现单例方法


DataSource.h
#import <Foundation/Foundation.h>

@interface DataSource : NSObject
@property (nonatomic, strong) NSString *myName;
+(DataSource*)sharedDataSource;
@end


DataSource.m
#import "DataSource.h"

@implementation DataSource
+(DataSource *)sharedDataSource{
    static DataSource *dataSource = nil;
    static dispatch_once_t once;
    dispatch_once(&once, ^{
        dataSource = [DataSource new];
    });
    return dataSource;
}
@end


(2)通过单例方式传值(感觉不太适合此类传值,如果要用单例方式传值的话,可以考虑此方式)

- (IBAction)singletonMethod:(id)sender {
    if ([self notEmpty]) {
        DataSource *dataSource = [DataSource sharedDataSource];
        dataSource.myName = self.nameTextField.text;
        [self dismissViewControllerAnimated:YES completion:nil];
    }else{
        [self showAlert];
    }
}

(3)通过单例方式取值

-(void)viewDidAppear:(BOOL)animated{
    [super viewDidAppear:animated];
    // 单例的办法进行传值
    DataSource *dataSource = [DataSource sharedDataSource];
    if ([dataSource.myName length] != 0) {
        self.nameLabel.text = dataSource.myName;
        // 置空,下一次填写值时候再用
        dataSource.myName = @"";
    }
}
上一篇下一篇

猜你喜欢

热点阅读