4

我创建了一个库,可以下载 JSON 数据,然后将其放入 NSDictionary。我用一个简单的 Twitter 引擎包装这个类,它允许我拉我朋友的时间线,发布更新并发布带有我的 GPS 位置的更新。根据我对 Objective-C 的有限经验,连接一切的方法是委托。我设置了一个委托属性,它将异步结果回调到选择器或方法签名。我什至可以在委托上创建一个可选或必需的接口,这将允许 Xcode 帮助我实现委托。为了了解如何在 Objective-C 中使用委托,我创建了这个简单的项目。

http://www.smallsharptools.com/downloads/ObjC/Delegates.zip

它定义了一个 Worker 类,它允许您使用委托初始化该类。当使用 doWork 方法完成工作时,它会在委托上查找方法签名以将消息发送回给它。它使用以下代码。

if([[self delegate] respondsToSelector:@selector(workFinished:)]) {
    NSString *msg = @"That's it? Easy!";
    [[self delegate] workFinished:msg];
}

它寻找 workFinished: 方法来传回消息。我将此方法签名声明为可选接口,并在标头 Worker.h 中使用以下代码。

  @protocol WorkerNotifications
   @optional
   - (void) workFinished: (NSString *) msg;
   @end

您可以从下载中查看项目的其余部分以获取所有详细信息。但是这两个代码片段展示了这种委托模式是如何工作的。但是对于 Twitter 类,我需要知道启动异步操作的方法的上下文,该操作会导致对委托方法的回调。如果我从调用类中多次调用 sendUpdate 方法,我应该如何知道回调的上下文?

通常使用 JavaScript、Java 或 C# 之类的语言,我会创建一个可以访问起始上下文的内联闭包或匿名类,但目前 iPhone 上的 Objective-C 不可能。我发现这个问题已经在 StackOverflow 上提出并回答了。

Objective-C 中的匿名委托实现?

所以我所做的是跳过可选接口,而是传入一个选择器,当异步操作完成时 Twitter 类将调用该选择器。开始此操作的调用看起来像...

CMTwitterEngine *engine = [[CMTwitterEngine alloc] initWithDelegate:self];
[engine setSendUpdateFinished:@selector(sendUpdateFinished:)];
[engine setSendUpdateFailed:@selector(sendUpdateFailed:)];
[engine setParsingSendUpdateFailed:@selector(parsingSendUpdateFailed:)];
[engine setUsername:TWITTER_USERNAME pass:TWITTER_PASSWORD];
[engine sendUpdate:statusUpdateText.text];

此代码首先使用 self 作为委托初始化引擎引用。为了附加我在选择器中发送的回调,这些选择器最初在 sendUpdate 方法签名上,但方法调用变得很长。我选择简单地设置选择器的属性。这一切都有效,但我不确定我是否喜欢它的工作方式,因为它只能部分解决我的问题。

为了完成这个示例,我完成了异步工作并最终在内部调用了一个方法,该方法查找给定的选择器并在它被定义时调用它。

- (void)sendUpdateFinished:(NSDictionary *)dictionary {
    if (self.sendUpdateFinished != nil) {
        [self.delegate performSelector:self.sendUpdateFinished withObject:dictionary];
    }
}

我可以传入状态消息以作为 Twitter 更新发送,但我仍然没有发起呼叫的上下文。如果我想多次调用 sendUpdate 并且第一个异步调用仍在运行怎么办?如果第二个电话先完成怎么办?他们都将 self 作为委托,因此我必须以某种方式跟踪上下文或将它们传递给不同的选择器以区分它们,这也不能满足我的需求。如果我有 3 或 4 或 5 个异步调用会发生什么?我需要知道哪些已成功发送以及何时完成。

看来我能做到这一切的唯一方法是创建一个类,该类包含上下文所需的所有属性,让该类充当调用异步 Twitter 方法的委托,然后向父级报告可能是 UIViewController 的类。我会采用这种方法,但我还没有阅读过这种方法,也没有看到任何这样做的示例代码。

你会怎么办?您将如何处理多个异步调用,这些调用可能以不同于外出的顺序结束,然后在完成时使用上下文处理它们?

4

3 回答 3

3

我认为您的情况是使用 NSNotificationCenter 的好地方

http://developer.apple.com/iphone/library/documentation/Cocoa/Reference/Foundation/Classes/NSNotificationCenter_Class/Reference/Reference.html

于 2010-02-03T17:33:10.480 回答
2

我必须第二(或第三)之前在 NSNotificationCenter 中发布的答案可能是您在这里寻找的。

本质上,当可能有许多委托人都需要做某事以响应已发生的单个操作或事件时,通常会使用通知。将其视为一对多委托或观察者模式的实现。需要了解的基本知识是:

  1. NSNotifications 有一个您定义的名称,它只是一个 NSString。通知可以按名称发布,对象注册以按名称接收通知。

  2. 发布通知时,可以提供 notificationSender 对象和/或 userInfo 字典。notificationSender 是在接收者处理给定通知时确定谁发布了给定通知的直接方法。userInfo 是一个 NSDictionary,可用于提供附加的上下文信息以及通知。

因此,与其强迫所有工作人员采用非正式协议并在运行时使用反射风格的调用方法,不如在 NSNotificationCenter 中注册工作人员的实例。通常,向 NSNotificationCenter 的注册是在每个工作类的 init 方法中完成的。然后,每种类型的工作人员的实例通常被设置为 NIB 中的“冻干”对象,或者可以在应用程序委托中以编程方式实例化,以便它们在应用程序生命周期的早期向通知中心注册。

事情发生时,您将 NSNotification 发布到 NSNotificationCenter (本质上是一个单例),然后注册接收该特定类型通知的所有其他内容都将具有指定用于处理该类型通知的方法。完成后,这些方法可以在发送者上调用一个通用方法(通过 NSNotification 的对象方法获得)来告诉发送者他们已经完成了他们的工作。

一旦每个已知的工作人员都签入了发送方的通用方法,就可以继续执行要执行的任何工作人员完成后代码。

于 2010-02-03T19:24:21.430 回答
1

要考虑的一件事是使用通知。简化代码,不那么紧密地耦合事物。

于 2010-02-03T17:30:25.433 回答