方法汇总

1、属性
2、方法
3、Delegate代理
4、Block回调
5、通知
6、KVO
7、单例
8、NSUserDefault(等本地化手段)

几种方式有着各自应用的场景,各有各的优点和缺点。


一、属性传值

使用场景:AtoB

说明:这种方式通常需要B对象,在A中设置B的属性,达到传值的目的。

使用:

在B中声明属性:

@property (copy ,nonatomic) NSString *value;

在A中修改B的属性

BCtrl *ctrl = [[BCtrl alloc] initWithNibName:@"BCtrl" bundle:nil];
ctrl.value = @"LOLITA";
[self.navigationController pushViewController:ctrl animated:YES];

这样在B中就可以使用这个value了。


二、方法传值

使用场景:AtoB

说明:这种方式也需要B对象,在A中调用B的方法并将数据传递给B

使用:

在B中声明方法

-(void)passValue:(NSDictionary *)dic;

在A中调用B的方法

[B_Object passValue:@{@"name":@"LOLITA"}];

B中使用传递过来的数据

-(void)passValue:(NSDictionary *)dic{
    NSLog(@"%@",dic);
}

效果:

body传值 ios ios传值方式_数据


使用场景:BtoA

说明:思路跟AtoB一样,使用A对象调用其方法传递数据

首先在A中定义传值方法并实现该方法

.h 文件中 声明该方法
@interface ACtrl : UIViewController
-(void)passValue:(NSDictionary *)dic;   // 传值方法
@end
.m 文件中 实现该方法
-(void)passValue:(NSDictionary *)dic{   // 使用传递过来的字典
    NSLog(@"%@",dic);
}

在B中声明一个id类型属性

@property (weak ,nonatomic) id delegate;    // 用来获得A对象,注意不要引起循环引用,我这里使用了weak

在A中初始化B的时候将A对象传递给B的这个属性

BCtrl *ctrl = [[BCtrl alloc] initWithNibName:@"BCtrl" bundle:nil];
ctrl.delegate = self;   // 指向id属性
[self.navigationController pushViewController:ctrl animated:YES];

在B的某个特定事件下(如返回),调用A的传值方法将数据传递过去

- (IBAction)goBackBtn:(UIButton *)sender {    // 返回事件
    if (self.delegate&&[self.delegate respondsToSelector:@selector(passValue:)]) {
        [self.delegate passValue:@{@"name":@"LOLITA",@"age":@"24"}];  //传递数据
    }
    [self.navigationController popViewControllerAnimated:YES];
}

这时候,A中的传值方法被调用,输出了B传过来的数据

body传值 ios ios传值方式_body传值 ios_02

在上面这个例子中,有人发现这样的使用和delegate很像,没错,delegate本来就是某个对象委托去达到某个目的而产生的设计模式,我这里使用A的传值方法来完成传值功能,那么我们iOS的设计模式中,完全可以将部分业务功能抽离出来,赋予代理,让其去实现这种功能。


三、Delegate代理传值

使用场景:BtoA

说明:代理模式是我们常用的设计模式,常用来委托完成某些业务功能,我们这里仅用来传递数据

首先我们新建一个协议文件,赋予其某种功能

#import <Foundation/Foundation.h>
@protocol PageDelegateProtocol <NSObject>
// 这里没有实现,只有方法声明,有些类似java中的接口
-(void)passValue:(NSDictionary *)dic;   // 具有传值功能的方法
@end

在B中导入该协议,并新建代理对象作为其属性

#import "PageDelegateProtocol.h"
@interface BCtrl : UIViewController
@property (nonatomic,weak) id <PageDelegateProtocol> delegate;  // 代理对象
@end

在B的某个特定事件下(如返回),将数据传递给A

- (IBAction)goBackBtn:(UIButton *)sender {
    if (self.delegate&&[self.delegate respondsToSelector:@selector(passValue:)]) {
        //使用协议中的传值方法
        [self.delegate passValue:@{@"name":@"LOLITA",@"age":@"24"}];
    }
    [self.navigationController popViewControllerAnimated:YES];
}

在A中,将A对象指向B中的代理

BCtrl *ctrl = [[BCtrl alloc] initWithNibName:@"BCtrl" bundle:nil];
ctrl.delegate = self;
[self.navigationController pushViewController:ctrl animated:YES];

A中遵守协议,并实现协议方法

-(void)passValue:(NSDictionary *)dic{
    NSLog(@"%@",dic);
}

这样我们就使用了代理实现了页面的传值,我们可以通过和BtoA的方法传值方式的对比来理解代理的方式


四、Block回调传值

block可以作为属性传值、也可以作为参数传值

1、block作为属性来传值

使用场景:BtoA

使用:

首先在B页面声明一个block作为属性

typedef void(^passValue)(NSDictionary *dic); // 声明一个block
@interface ACtrl : UIViewController
@property (nonatomic,copy) passValue passDic;  // 作为B的属性
@end

在特定事件(如返回事件)使用block回调

- (IBAction)goBackBtn:(UIButton *)sender {
    if (self.passDic) {
       self.passDic(@{@"name":@"LOLITA",@"age":@"24",@"sex":@"male"});  // 回调数据
    }
    [self.navigationController popViewControllerAnimated:YES];
}

在A中,实现block回调函数

BCtrl *ctrl = [[BCtrl alloc] initWithNibName:@"BCtrl" bundle:nil];
ctrl.passDic = ^(NSDictionary *dic) { // 实现block
        NSLog(@"%@",dic);
    };
[self.navigationController pushViewController:ctrl animated:YES];

2、block作为参数来传值

使用场景:BtoA

说明:这种方式在请求类中比较常见,在请求回来数据之后使用block回调数据,这里使用两个页面简单介绍使用过程

先在B类中,定义一个block

typedef void(^passValue)(NSDictionary *dic);

将bock作为参数声明一个方法

-(void)passDic:(passValue)dicBlock;

实现该方法

-(void)passDic:(passValue)dicBlock{
    if (dicBlock) {
        // 使用block回调数据
        dicBlock(@{@"name":@"LOLITA",@"age":@"24"});
    }
}

在A类中,使用B类调用带block参数的方法

BCtrl *ctrl = [[BCtrl alloc] initWithNibName:@"BCtrl" bundle:nil];
[ctrl passDic:^(NSDictionary *dic) {
    NSLog(@"--%@",dic);
}];

五、通知传值

使用场景:Ato多个页面

说明:通知属于观察者模式,属于一对多的传递消息,相比于代理和block的一对一,通知的开销较大,所有只有在需要将消息传递给多个页面的时候再使用通知,另外通知属于单例

使用:

在需要接收通知消息的页面添加注册

// 给self添加通知事件,当name和发送方相同的时候才能接受到消息
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(getUserInfo:) name:@"passValue" object:nil];

// 处理消息的方法
-(void)getUserInfo:(id)userInfo{
    NSLog(@"%@",userInfo);
}

传递消息的页面,特定的事件发送通知(如返回)

- (IBAction)goBackBtn:(UIButton *)sender {
    [self.navigationController popViewControllerAnimated:YES];
    [[NSNotificationCenter defaultCenter] postNotificationName:@"passValue" object:nil userInfo:@{@"name":@"LOLITA",@"sex":@"male"}];
}

是不是非常方便呢?还需要注意的是:1、注册通知一定要在发送通知的代码执行之前 2、通知需要手动移除

// 在添加监听的页面移除通知,这里移除了name为passValue的通知,也可以不区分name全部移除
-(void)dealloc{
    [[NSNotificationCenter defaultCenter] removeObserver:self name:@"passValue" object:nil];
}

六、KVO (Key-Value Observing)键值观察

KVO顾名思义键值对观察,它通过监听被观察者的某些属性值的变化而响应某些事件,也是一种消息传递。

为了演示,我给B页面设置一个属性,作为被观察的属性

@interface BCtrl : UIViewController
@property (assign ,nonatomic) NSInteger value;
@end

使用:

首先,被观察者需要为自己被监听的属性添加监听事件,B是被观察者

// A页面中
- (IBAction)btn:(UIButton *)sender {
    BCtrl *ctrl = [[BCtrl alloc] initWithNibName:@"BCtrl" bundle:nil];
    // self是A页面,它监听B页面的value属性 ,options有多种,这里获取新值
    [ctrl addObserver:self forKeyPath:@"value" options:NSKeyValueObservingOptionNew context:@"changeValue"];
    [self.navigationController pushViewController:ctrl animated:YES];
}

观察者A,需要实现observeValueForKeyPath:ofObject:change:context:方法,来获取监听属性的变化

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
    if (context == @"changeValue") {
        if ([keyPath isEqualToString:@"value"]) {
            NSLog(@"%@",change);
            // 记得要移除观察者
            [object removeObserver:self forKeyPath:@"value"];
        }
    } else {
        [super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
    }
}

这样当被观察者B的value属性发生变化时,A就可以得到消息并获取相应的值

- (IBAction)goBackBtn:(UIButton *)sender {
    [self.navigationController popViewControllerAnimated:YES];
    self.value = 10;
}

效果

body传值 ios ios传值方式_body传值 ios_03



以上方法都是明确知道被传递的对象是谁,要做什么,属于一种消息的传递,是事件响应(属性传值可以通过重写setter方法来实现事件的响应),但是单例和NSUserDefault不同,它们不像代理明确需要做什么,也不像通知可以传递消息,仅仅被用来存储和获取数据,并且,它们不知道谁存储的数据,也不知道谁来调用,这些事情需要我们自己主动去做。



七、单例

单例即在app生命进程中仅被初始化一次,它的内存地址在初始化时就确定了,所以我们可以在A页面存储数据,B页面取数据,并且值是相同的

注意:当app生命进程结束后,单例的内存空间也被释放了,数据会消失掉

使用:

首先我们需要新建单例对象

.h 文件
#import <Foundation/Foundation.h>
@interface DataManager : NSObject
@property (strong ,nonatomic) NSDictionary *dic;  // 存储的数据
+(instancetype)shareDataManager;   // 初始化方法
@end
.m 文件
+(instancetype)shareDataManager{
    static dispatch_once_t onceToken;
    static DataManager *manager;
    dispatch_once(&onceToken, ^{
        manager = [DataManager new];
    });
    return manager;
}

使用单例存数据,在需要的地方导入单例文件

// 存储一个字典数据
[DataManager shareDataManager].dic = @{@"name":@"LOLITA",@"iOS":@"2"};

在需要取数据的地方

NSLog(@"%@",[DataManager shareDataManager].dic);

效果

body传值 ios ios传值方式_ios_04


八、NSUserDefault等本地化手段

NSUserDefault是使用KVC手段访问、修改数据,并会在本地持久化

存储数据:通过不同的key来存数据

[[NSUserDefaults standardUserDefaults] setObject:@"LOLITA" forKey:@"name"];
[[NSUserDefaults standardUserDefaults] synchronize];

获取数据:key要相同

[[NSUserDefaults standardUserDefaults] objectForKey:@"name"]

删除数据:

[[NSUserDefaults standardUserDefaults] removeObjectForKey:@"name"];

单例和NSUserDefault操作数据比较

单例的存储的数据在app结束时就消失了,NSUserDefaults属于本地化存储,存储的数据只要app不删除,就一直存在,所以在使用这两种方式存取数据时要根据业务需求来。


九、总结

自己体会吧。。。啊哈哈哈