从多个视图控制器接收委托方法调用

时间:2015-05-10 09:16:34

标签: ios swift uiviewcontroller delegates uisplitviewcontroller

我有一个SplitViewController的应用。 MasterViewControllerUITableViewControllerDetailViewControllerUIViewControllerMapView

enter image description here

我正在调用API来获取事件列表。在tableView中,我想显示事件的名称和地址。在mapView中,我想标记这些事件的位置。

我有一个名为DataLoader的类,我在其中调用API,解析JSON对象并创建Event个对象并将它们成功放入数组中。

import Foundation
import SwiftyJSON

@objc protocol DataLoaderDelegate {
    func eventDataReceived(items: [Event]?, error: NSError?)
}

public class DataLoader {

    private let api = ApiClient()

    var delegate: DataLoaderDelegate?

    init() { }

    public func fetchEvents() {
        api.getEvents({ (data) -> Void in
            let json = JSON(data)
            self.processEventData(json["data"])
        }, failure: { (error) -> Void in
            println("Error fetching events: \(error?.localizedDescription)")
            self.delegate?.eventDataReceived(nil, error: error)
        })
    }

    private func processEventData(data: JSON) {
        var events = [Event]()

        if let eventsArray = data.array {
            for eventObj in eventsArray {
                let event = Event(
                    id: eventObj["id"].int!,
                    type: EventType(rawValue: eventObj["type"].int!)!,
                    location: eventObj["location"].string!,
                    status: EventStatus(rawValue: eventObj["status"].int!)!
                )
                event.allDay = eventObj["allDay"].int!
                event.title = eventObj["title"].string
                event.description = eventObj["description"].string
                event.latitude = eventObj["lat"].double
                event.longitude = eventObj["lng"].double
                event.startDate = NSDate(string: eventObj["start"].string!)
                event.endDate = NSDate(string: eventObj["end"].string!)
                event.createdAtDate = NSDate(string: eventObj["created_at"].string!)
                event.updatedAtDate = NSDate(string: eventObj["updated_at"].string!)
                event.creatorEmail = eventObj["creatorEmail"].string
                event.organizerEmail = eventObj["organizerEmail"].string
                event.googleCalendarId = eventObj["google_cal_id"].string!
                event.googleEventId = eventObj["google_event_id"].string!
                event.htmlLink = eventObj["htmlLink"].string!

                events.append(event)
            }

            // Order the events by the startDate value
            events.sort({ $0.startDate!.timeIntervalSince1970 < $1.startDate!.timeIntervalSince1970 })

            self.delegate?.eventDataReceived(events, error: nil)
        }
    }
}

我有一个名为DataLoaderDelegate的委托,其方法名为func eventDataReceived(items: [Event]?, error: NSError?),在将所有事件对象插入数组后调用,以便我可以通过该方法传递它。

我需要在填充tableView的ListViewController和填充mapView的MapViewController中实现此方法。所以我在两个视图控制器中都实现了eventDataReceived委托方法,以获取事件对象。

ListViewController.swift

import UIKit

class ListViewController: UITableViewController, DataLoaderDelegate {

    let loader = DataLoader()

    private var events = [Event]()

    override func viewDidLoad() {
        super.viewDidLoad()

        loader.delegate = self
        loader.fetchEvents()
    }

    // MARK: - Table view data source
    override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return events.count
    }

    override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCellWithIdentifier("Cell", forIndexPath: indexPath) as! UITableViewCell

        let event = events[indexPath.row]
        cell.textLabel?.text = event.title
        cell.detailTextLabel?.text = event.location

        return cell
    }

    // MARK: - DataLoaderDelegate
    func eventDataReceived(items: [Event]?, error: NSError?) {
        println("ListViewController got \(items?.count) events")
        if let events = items {
            self.events = events
            tableView.reloadData()
        }
    }
}

MapViewController.swift

import UIKit

class MapViewController: UIViewController, DataLoaderDelegate {

    let loader = DataLoader()

    override func viewDidLoad() {
        super.viewDidLoad()

        loader.delegate = self
    }

    // MARK: - DataLoaderDelegate
    func eventDataReceived(items: [Event]?, error: NSError?) {
        println("MapViewController got \(items?.count) events")
    }
}

但这是问题所在。只有ListViewController中的实现才会被执行。我认为原因是我在DataLoader中创建了MapViewController类的新实例。

如何在整个应用程序中拥有一个DataLoader实例并从所有视图控制器接收其委托方法调用?

我已经将一个演示项目上传到我的Dropbox,如果你需要运行它以获得更好的想法。

感谢。

1 个答案:

答案 0 :(得分:2)

这是Singleton-pattern的最佳做法案例 你的DataLoader应该实现一个静态let,它保存对self的同一个对象的引用。

static let sharedInstance = DataLoader()

当你想要引用不断相同的单例对象时。使用DataLoader.sharedInstance

进行调用

确保始终使用.sharedInstance,并且不要忘记擦除类的标准初始化程序的任何调用,因为除非您以编程方式阻止此行为,否则它仍将生成新实例。

<强>解释

只插入单例常量(如上所示)不会使DataLoader类始终返回单例实例。现有的初始化程序调用如下:

   var myDataLoader = DataLoader()
每次调用它们时,

仍然会实例化一个新对象。通过以下方式达到单例实例:

   var mySingletonDataLoader = DataLoader.sharedInstance

您可以更改标准(非单身)初始化程序,如:

init() {
NSLog("No instances allowed, please use .sharedInstance for a singleton")
}

完整解决方案

单身只是解决整个问题的一部分。使用的委托模式非常适合将信息从一个对象发送到另一个对象,从而委派工作。 提问者的问题需要从一个对象到许多其他对象的另一种广播机制。因此,他决定使用Observer-pattern

实施NSNotificationCenter