【问题标题】:Passing data with a delegate from a closure使用闭包中的委托传递数据
【发布时间】:2018-02-03 22:12:02
【问题描述】:

我有两个视图控制器,我正在尝试将数据从一个传递到另一个。数据从 MKLocalSearch 闭包返回。但我似乎无法让我的委托方法运行。我希望有人能对此有所了解吗?我模拟了我正在尝试做的一个较小的版本。另外,我不使用故事板。我把所有东西都编码了。这是代码...

import UIKit
import MapKit

protocol SendDataDelegate {
    func sendData(data: String)
}

class OneViewController: UIViewController {

var delegate: SendDataDelegate?

override func viewDidLoad() {
    super.viewDidLoad()

    doSearch() { coord in
        if let coord = coord {
            //When the execution gets here, coord does have
            //the values to be sent to the nexr view controller.
            self.delegate?.sendData(data: "\(coord)")
            let twoViewController = TwoViewController()
            self.present(twoViewController, animated: true)
        }
    }
}

func doSearch(completion: @escaping (CLLocationCoordinate2D?) -> Void) {

    var coord: CLLocationCoordinate2D?
    let request = MKLocalSearchRequest()

    request.naturalLanguageQuery = "New York"

    let search = MKLocalSearch(request: request)

    search.start(completionHandler: {(response, error) in

        if error != nil {
            print("Error occured in search:\(error!.localizedDescription)")
        } else if response!.mapItems.count == 0 {
            print("No matches found")
        } else {
            print("Matches found")

            coord = response?.mapItems[0].placemark.coordinate
        }
        completion(coord)
    })
  }
}

import UIKit

class TwoViewController: UIViewController, SendDataDelegate {

var myData: String = ""
var oneViewController = OneViewController()

override func viewDidLoad() {
    super.viewDidLoad()
    oneViewController.delegate = self
}

func sendData(data: String) {
    myData = data
    print ("myData: \(myData)")
}

}

【问题讨论】:

    标签: delegates closures swift4 mklocalsearch


    【解决方案1】:

    在您的TwoViewController 中,您有一个属性,其中包含OneViewController另一个 实例,并将它的委托设置为self

    因此,每次创建TwoViewController 的实例时,都会创建一个OneViewController 的新实例,但由于显示OneViewController 的实际视图,它们实际上都不是那个实例。

    在你的情况下,委托模式不合适,你最好直接调用TwoViewController的方法:

    class OneViewController: UIViewController {
    
        //no need to have a delegate
    
        override func viewDidLoad() {
            super.viewDidLoad()
    
            doSearch() { coord in
                if let coord = coord {
                    //When the execution gets here, coord does have
                    //the values to be sent to the nexr view controller.
                    let twoViewController = TwoViewController()
                    twoViewController.sendData(data: "\(coord)")
                    self.present(twoViewController, animated: true)
                }
            }
        }
    
        //...    
    }
    
    class TwoViewController: UIViewController {
    
        var myData: String = ""
        //Creating a new instance of `OneViewController` has no meaning.
    
        override func viewDidLoad() {
            super.viewDidLoad()
            //Useless to set the delegate of newly created `OneViewController`.
        }
    
        func sendData(data: String) {
            myData = data
            print ("myData: \(myData)")
        }
    
    }
    

    如果您有任何理由在TwoViewController 中拥有属性oneViewController,请解释原因。您当前的代码没有任何解释。

    【讨论】:

    • 您对参考的看法是绝对正确的。我休息了一下。回来看到错误的参考。我现在的委托模式可以正常工作了!谢谢。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-11-17
    • 2017-01-25
    • 2013-01-20
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多