博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
IOS笔记047-代理传值和block传值
阅读量:5943 次
发布时间:2019-06-19

本文共 4693 字,大约阅读时间需要 15 分钟。

在两个不同的控制器之间传递数据,可以使用代理传值或者block传值。

例子是一个简单通讯录。

主界面如下:

屏幕快照 2015 06 13 13 05 15

添加联系人界面

屏幕快照 2015 06 13 13 05 08

查看/编辑联系人界面:默认是查看模式,点击编辑后进入编辑模式

屏幕快照 2015 06 13 13 05 23

编辑模式

屏幕快照 2015 06 13 13 05 36

数据更新成功。

屏幕快照 2015 06 13 13 05 44

其中添加联系人界面的数据传递使用代理方式实现。

编辑联系人界面的数据传递使用block实现。

下面来看具体过程

1、整个界面搭建

屏幕快照 2015 06 13 13 32 34

在storyboard里拖拽四个控制器,其中联系人界面是一个UITableView。界面之间的跳转使用代码实现,但是要给每一个控制器指定一个标识。按功能分别指定为login,contact,add,edit.

屏幕快照 2015 06 13 13 25 46   屏幕快照 2015 06 13 13 37 10

具体细节我就不说了,关键是代理传值的实现。

2、代理传值

数据传递方向是从add控制器传递到contact控制器。至于为什么使用代理,主要是为了降低类之间的耦合度。

    1、这里需要给add控制器添加一个代理对象,然后定义一个代理需要遵守的协议。

    // 代理实现逆传数据

    @classSLQContact; // 模型对象

    @classSLQAddViewController; //

    @protocol SLQAddViewControllerDelegate<NSObject>

 

    @optional

    // 代理方法

    - (void)addViewController:(SLQAddViewController *)addVC DidClickBtnWithContact:(SLQContact *)contact;

 

    @end

 

    @interface SLQAddViewController : UIViewController

    /*代理对象*/

    @property (strong, nonatomic) id<SLQAddViewControllerDelegate> delegate;

 

    @end

    2、等需要传递数据时只需通知代理一声即可,不需要关心代理如何做。

在添加联系人界面里点击添加按钮后就去通知代理传递数据。

    // 添加联系人按钮,单击后传递数据到联系人控制器,并返回上一个界面

    - (IBAction)addBtn:(id)sender {

        

        //传递模型数据

        SLQContact *temp = [SLQContactcontactWithName:_nameField.textandPhone:_phoneField.text];

 

        // 通知代理

        if([_delegate respondsToSelector:@selector(addViewController:DidClickBtnWithContact:)])

        {

            [_delegateaddViewController:selfDidClickBtnWithContact:temp];

        }

        // 返回上一个界面

        [self.navigationControllerpopViewControllerAnimated:YES];

    }

    3、还有最关键的一步,那就是指定代理对象是谁

        这里指定代理对象的是contact控制器,因为,要把数据传递给它,所以它作为接收者也就是代理方。

    // 添加按钮,按下后进入添加联系人界面

    - (IBAction)addBtn:(id)sender {

        //

    //    [self performSegueWithIdentifier:@"contactToAdd" sender:nil];

        // 通过代码获取storyboard中得控制器

        UIStoryboard *story = [UIStoryboard storyboardWithName:@"Main" bundle:nil];

        // 获取main。storyboard中得控制器,以标识符区分

        SLQAddViewController* addVC = [story instantiateViewControllerWithIdentifier:@"add"];

        // 代理逆传数据

        //如果通过代码跳转的话,必须知道目的控制器,整个程序中也只有这个时候才知道下一个控制器是谁,所以在这里指定代理对象为contact控制器再好不过了

        addVC.delegate = self;

        // 跳转到添加联系人界面    [self.navigationController pushViewController:addVC animated:YES];

    }

    // 实现代理方法,记得让类遵守协议

    - (void)addViewController:(SLQAddViewController *)addVC DidClickBtnWithContact:(SLQContact *)contact

    {

        // 添加数据到数组

        [_contacts addObject:contact];

        

        // 刷洗表格

        [self.tableView reloadData];

    }

2、block传值

    1、在编辑控制器中对block进行生声明以及定义

    // blocl传值使用

    @classSLQContact;

    // 声明block别名,参数为要传递的数据。

    typedef void(^SLQEditViewControllerBolok)(SLQContact *);

 

    @interface SLQEditViewController : UIViewController

    /*模型*/

    @property (strong, nonatomic) SLQContact *contact;

    /*block 对象*/

    @property (strong, nonatomic) SLQEditViewControllerBolok block;

    @end

 

    2、在点击保存按钮后进行数据传递

    // 保存按钮事件

    - (IBAction)save:(id)sender

    {

        // 传递模型数据

        SLQContact *contact = [SLQContactcontactWithName:_nameField.textandPhone:_phoneField.text];

        // block实现传值,先检查是否有数据,如果有传递模型数据

        if(_block)

        {

            _block(contact);

        }

        // 回到上个界面

        [self.navigationControllerpopViewControllerAnimated:YES];

    }

    3、关键一点还是要在数据接收方也就是contact控制器中对block内容进行包装

      因为要通过代码跳转,同样需要知道目的控制器,跳转的地方就是选中cell的时候。

    // 选中cell后进入编辑界面

    - (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath

    {

        // 获取编辑控制器

        UIStoryboard *story  = [UIStoryboard storyboardWithName:@"Main" bundle:nil];

        // 获取目的控制器

        SLQEditViewController *editVC = [story instantiateViewControllerWithIdentifier:@"edit"];

        // 传递模型数据到edit控制器(这是一种顺序传值)

        editVC.contact = self.contacts[indexPath.row];

        // 使用block包装之后要进行的操作

        editVC.block = ^(SLQContact *contact){

            // 修改数据

            [self.contacts replaceObjectAtIndex:indexPath.row withObject:contact];

            // 刷新表格

            [self.tableView reloadData];

        };

        // 跳转到编辑界面

        [self.navigationControllerpushViewController:editVC animated:YES];

    }

3、顺序传值

顺序传递数据比价简单,只需要接收方有一个属性对要传递的数据进行接收就行。

上面在进入编辑控制界面时就要对数据进行传递,要把在联系人界面的数据传递到编辑控制器界面,然后对其进行修改。

    // 选中cell后进入编辑界面

    - (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath

    {

        // 获取编辑控制器

        UIStoryboard *story  = [UIStoryboard storyboardWithName:@"Main" bundle:nil];

        // 获取目的控制器

        SLQEditViewController *editVC = [story instantiateViewControllerWithIdentifier:@"edit"];

        // 传递模型数据到edit控制器(这是一种顺序传值)

        editVC.contact = self.contacts[indexPath.row];

        // 使用block包装之后要进行的操作

        editVC.block = ^(SLQContact *contact){

            // 修改数据

            [self.contacts replaceObjectAtIndex:indexPath.row withObject:contact];

            // 刷新表格

            [self.tableView reloadData];

        };

        // 跳转到编辑界面

        [self.navigationControllerpushViewController:editVC animated:YES];

    }

上面的代码这样写。是因为在编辑控制器中已经定义了一个模型对象接收传递的数据。

    @interface SLQEditViewController : UIViewController

    /*模型对象,接收传递过来的数据*/

    @property (strong, nonatomic) SLQContact *contact;

    /*block 对象*/

    @property (strong, nonatomic) SLQEditViewControllerBolok block;

    @end

同样传递的地方也是在控制器跳转之前进行数据的传递。

4、总结

传值的方是由到导航控制器的行走方向决定的。

顺序传值:

      由源控制器传递当目的控制器。

      接收方有一个属性接收传递数据,在控制器跳转之前进行传递

逆序传值:

      由源控制器传递当目的控制器。

      代理:

              在发送方对声明代理对象,然后定义代理协议,要传递的数据要放在代理方法的参数中,最后在触发事件后通知代理

              在接收方设置代理位接收方,实现代理方法

      block:

              在发送方对声明block,要传递的数据要放在block的参数中,最后在触发事件后调用block

              在接收方设置block的内容

 

转载于:https://www.cnblogs.com/chengzhengfu/p/4573926.html

你可能感兴趣的文章
并查集hdu1232
查看>>
改动Androidproject的名称(非Eclipse重命名)
查看>>
tomcat work目录的作用就是编译每个项目里的jsp文件为java文件如果项目没有jsp页面则这个项目文件夹为空...
查看>>
dedecms后台左侧菜单500错误怎么处理
查看>>
Maven配置将war包部署到Tomcat(tomcat7-maven-plugin)
查看>>
Spring MVC学习-------------訪问到静态的文件
查看>>
Unity应用架构设计(11)——一个网络层的构建
查看>>
运行自己的shell脚本
查看>>
内存错误的类别
查看>>
Authentication 方案优化探索(JWT, Session, Refresh Token, etc.)
查看>>
Struts2 关于返回type="chain"的用法.
查看>>
Maven私服安装及配置——(十二)
查看>>
设计模式 - 迭代器模式(iterator pattern) 具体解释
查看>>
Codeforces554B:Ohana Cleans Up
查看>>
【java】jvm查看当前虚拟机堆大小限制
查看>>
python写入excel(xlswriter)--生成图表
查看>>
NetworkStream.write只能使用一次,后面再使用无效
查看>>
oracle进行字符串拆分并组成数组
查看>>
100多个基础常用JS函数和语法集合大全
查看>>
Java8 lambda表达式10个示例
查看>>