【iOS 开发】Controller 之间使用代理传值

Controller 传值

控制器之间经常需要互相传递值,第一个控制器(简称 MasterVC)在通过 NavigationController Push 第二个控制器(简称 DetailVC)的时候,可以捕获到 DetailVC,所以可以设定后者的变量。而 DetailVC 在给 MasterVC 传递值的时候,比如设定 MasterVC 的 Title,却不能用同样的方式传值(试一下就知道了),而且因为我们要通过 NavigationController 返回原来的界面而不是 push 一个新的界面,所以也不能通过 segue 传值,解决方法我尝试了两种:

  1. 使用 static var
  2. 使用代理传值

使用 static var

class MasterVC: UIViewController {
    static var text:String! = ""
    override func viewWillAppear(animated: Bool) {
        title = MasterVC.text
    }
}

class DetailVC: UIViewController {
    override func viewWillDisappear(animated: Bool) {
        MasterVC.text = "xxx"
    }
}

这样虽然行得通,但是代码丑陋:类似 viewWillAppear 这样的方法,让人难以读懂,为什么这样的事情要在这个地方做,没有明显的道理;在 MasterVC 内部,仍然需要以 <code>MasterVC.text</code> 这样蹩脚的方式调用。

使用代理传值

说白了我们只是希望 MasterVC 有一个 <code>setControllerTitle(text:String)</code> 这样的方法,然后在 DetailVC 里面适时地调用这个方法就好了。于是解决方案是:将 MasterVC 设为 DetailVC 的代理,DetailVC 以代理作为桥梁,来调用 MasterVC 里面的 <code>setControllerTitle(text:String)</code> 方法,而 MasterVC 只需要遵守一个相应的传值协议,协议里面要求必须实现的方法,就是 <code>setControllerTitle(text:String)</code> 这个方法。

//  PassDataDelegate.swift

import Foundation

protocol PassDataDelegate {
    func setControllerTitle(text:String)
}


//  MasterVC.swift

import UIKit

class MasterVC: UIViewController, PassDataDelegate {
    
    @IBOutlet weak var textField: UITextField!
    
    override func viewDidLoad() {
        super.viewDidLoad()
    }
    
    
    // MARK: - Pass Data
    @IBAction func goNextController(sender: AnyObject) {
        guard let vc = storyboard?.instantiateViewControllerWithIdentifier("DetailVC")
            as? DetailVC else { return }
        guard let txt = textField.text else { return }
        vc.title = txt // pass data
        vc.delegate = self // get data by set delegate
        navigationController?.pushViewController(vc, animated: true)
    }
    

    // MARK: - PassDataDelegate Method
    func setControllerTitle(text:String) {
        title = text
    }

}


//  DetailVC.swift

import UIKit

class DetailVC: UIViewController {
    
    @IBOutlet weak var textField: UITextField!
    var delegate: PassDataDelegate!
    
    override func viewDidLoad() {
        super.viewDidLoad()
    }
    
    
    // MARK: - Pass Data
    @IBAction func goBackLastController(sender: AnyObject) {
        if delegate != nil {
            guard let txt = textField.text else { return }
            delegate.setControllerTitle(txt)
        }
        navigationController?.popViewControllerAnimated(true)
    }
    
}

相比之前的方法,我们获得的额外的好处是:假如我们传递的不只是字符串,而是大量的信息,比如十个不同类型的值,使用代理传值依然只需要实现一个方法,而在之前的解决方案中,我们需要写十行 <code>static var</code>。

完整项目

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,828评论 19 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 32,379评论 18 399
  • importUIKit classViewController:UITabBarController{ enumD...
    明哥_Young阅读 9,463评论 1 10
  • 1.badgeVaule气泡提示 2.git终端命令方法> pwd查看全部 >cd>ls >之后桌面找到文件夹内容...
    i得深刻方得S阅读 10,249评论 1 9
  • 一见客户就紧张,不知道说什么。 想要与客户打交道,快速成为好朋友——怎么办? ...【快速找到共鸣话题是打开友谊之...
    智盛心法阅读 1,238评论 0 0

友情链接更多精彩内容