[转]iOS页面间传值的方式(Delegate/NSNotification/Block/NSUserDefault/单例)

iOS页面间传值的方式(NSUserDefault/Delegate/NSNotification/Block/单例)

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

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

情况1: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];  
}  

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

主流方案:

(1)通过委托delegate的方式实现

image.png

设置协议及方法

//SecondViewController.h  
  
@protocol secondViewDelegate  
-(void)showName:(NSString *)nameString;  
@end  

设置代理(为防止循环引用,此处采用了weak)

//SecondViewController.h  
  
@interface SecondViewController : UIViewController  
@property (nonatomic, weak)id<secondViewDelegate> delegate;  
@property (nonatomic, copy) ablock block;  
@end  

调用,显示

//RootViewController.m  
-(void)showName:(NSString *)nameString{  
    self.nameLabel.text = nameString;  
} 

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

(2)通过通知notification的方式实现

image.png

在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];  
    }  
}  

在A页面的控制器中,注册通知:

//RootViewController.m  
- (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];  
}

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

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

调用,显示

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

(3)block方式实现

block介绍:http://blog.csdn.net/totogo2010/article/details/7839061

链接一篇描述block回调挺有意思的文章 http://blog.csdn.net/mobanchengshuang/article/details/11751671

分析:

在B视图控制器中,定义一个block,参数为字符串

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

在B视图控制器中,当输入名字,点击对应的确定按钮后

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

在A视图显示,回调block

- (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;  
    };  
}  

(4)KVO方式实现

KVO实现原理介绍:http://blog.csdn.net/kesalin/article/details/8194240

在A视图中,编写以下代码

//A视图  
//一个指向B视图的成员变量  
@property (nonatomic, strong) SecondViewController *second;  
  
//在A视图跳转到B视图的地方添加如下代码  
    self.second = [[SecondViewController alloc] initWithNibName:@"SecondViewController" bundle:nil];  
    [self.second addObserver:self forKeyPath:@"userName" options:NSKeyValueObservingOptionNew context:nil];  
    [self presentViewController:self.second animated:YES completion:nil];  
  
  
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(voidvoid *)context  
{  
//此处监听key对应值的变化情况  
    if ([keyPath isEqualToString:@"userName"]) {  
        self.myLabel.text = self.second.userName;  
    }  
}  
  
//清理观察  
- (void)dealloc  
{  
    [self.second removeObserver:self forKeyPath:@"userName"];  
}  

在B视图编写以下代码

//在B视图  
//.h文件  
  
@property (nonatomic, strong) NSString *userName;//待监听的成员变量  
  
//可以在两处修改userName的值。一个是设置textfield的UITextFieldDelegate。实现一下方法  
  
-(void)textFieldDidEndEditing:(UITextField *)textField{  
self.userName = self.myField.text;  
}  
  
//或者在B视图,点击确定按钮,跳转回A视图的时候,修改userName的值也可以  
  
- (IBAction)buttonPressed:(id)sender {  
    self.userName = self.myField.text;  
    [self dismissViewControllerAnimated:YES completion:nil];  
} 

在查阅资料的过程中,我还看到了以下几种方案:
(1)使用SharedApplication,定义一个变量来传递(感觉和单例的方式一样)
(2)使用文件,或者NSUserdefault来传递

//通过文件或者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];  
    }  
} 

在A视图控制器显示

-(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"]; 
    } 
    DataSource *dataSource = [DataSource sharedDataSource]; 
    if ([dataSource.myName length] != 0) { 
        self.nameLabel.text = dataSource.myName; 
        dataSource.myName = @""; 
    } 
*/  
}  

(3)通过一个单例的class来传递
B视图控制器

//通过单例方式传值(感觉不太适合此类传值,如果要用单例方式传值的话,可以考虑此方式)  
- (IBAction)singletonMethod:(id)sender {  
    if ([self notEmpty]) {  
        DataSource *dataSource = [DataSource sharedDataSource];  
        dataSource.myName = self.nameTextField.text;  
        [self dismissViewControllerAnimated:YES completion:nil];  
    }else{  
        [self showAlert];  
    }  
}  

A视图控制器显示

-(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"]; 
    } 
    DataSource *dataSource = [DataSource sharedDataSource]; 
    if ([dataSource.myName length] != 0) { 
        self.nameLabel.text = dataSource.myName; 
        dataSource.myName = @""; 
    } 
*/  
}  
@end  

这里面用到了单例模式,编写了DataSource这个类,存放数据

//  
//  DataSource.h  
//  TestCallBack  
//  
//  Created by csdc-iMac on 14-7-17.  
//  Copyright (c) 2014年 JuneWang. All rights reserved.  
//  
  
#import <Foundation/Foundation.h>  
  
@interface DataSource : NSObject  
@property (nonatomic, strong) NSString *myName;  
+(DataSource*)sharedDataSource;  
@end 
//  
//  DataSource.m  
//  TestCallBack  
//  
//  Created by csdc-iMac on 14-7-17.  
//  Copyright (c) 2014年 JuneWang. All rights reserved.  
//  
  
#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  

程序运行截图

A视图:


image.png

B视图:


image.png

当输入姓名,并点击对应的确认按钮后,会回到A视图,并显示在B视图中输入的姓名:

image.png

PS:用全局变量的方式也可以实现页面传值的效果。


源码地址:https://github.com/wangtao169447/PassValue

转自师父博客:http://www.cnblogs.com/JuneWang/p/3850859.html

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容