如何在Objective中创建委托

我知道代表如何工作,并且我知道我可以如何使用它们。

但是,我如何创建它们?


Objective-C委托是已经分配给delegate属性另一个对象的对象。 要创建一个,您只需定义一个实现您感兴趣的委托方法的类,并将该类标记为实现委托协议。

例如,假设你有一个UIWebView 。 如果你想实现它的委托的webViewDidStartLoad:方法,你可以像这样创建一个类:

@interface MyClass<UIWebViewDelegate>
// ...
@end

@implementation MyClass
- (void)webViewDidStartLoad:(UIWebView *)webView { 
    // ... 
}
@end

然后,您可以创建一个MyClass的实例,并将其作为Web视图的委托进行分配:

MyClass *instanceOfMyClass = [[MyClass alloc] init];
myWebView.delegate = instanceOfMyClass;

UIWebView方面,它可能具有类似于此的代码,以查看委托是否响应使用respondsToSelector:webViewDidStartLoad:消息respondsToSelector:并在适当时发送它。

if([self.delegate respondsToSelector:@selector(webViewDidStartLoad:)]) {
    [self.delegate webViewDidStartLoad:self];
}

委托属性本身通常被声明为weak (在ARC中)或assign (在ARC之前)以避免保留循环,因为对象的委托通常持有对该对象的强引用。 (例如,视图控制器通常是它包含的视图的代表。)

为您的类制作代表

为了定义你自己的委托,你必须在某个地方声明它们的方法,正如Apple Docs on protocols所讨论的那样。 你通常会声明一个正式的协议。 声明,从UIWebView.h转述,看起来像这样:

@protocol UIWebViewDelegate <NSObject>
@optional
- (void)webViewDidStartLoad:(UIWebView *)webView;
// ... other methods here
@end

这与接口或抽象基类相似,因为它在这种情况下为您的委托UIWebViewDelegate创建了一个特殊类型。 代表实现者必须采用这个协议:

@interface MyClass <UIWebViewDelegate>
// ...
@end

然后在协议中实现这些方法。 对于在协议中为声明的方法@optional (最喜欢的委托方法),您需要检查-respondsToSelector:调用它特定的方法之前。

命名

委托方法通常以委派​​类名称开始命名,并将委托对象作为第一个参数。 他们也经常使用遗嘱,遗嘱或遗嘱。 因此, webViewDidStartLoad:webViewDidStartLoad:第一个参数是web视图),而不是loadStarted (不带参数)。

速度优化

每次我们要发送消息时,不要检查代理是否响应选择器,而是可以在代理设置时缓存该信息。 一个非常干净的方法是使用一个位域,如下所示:

@protocol SomethingDelegate <NSObject>
@optional
- (void)something:(id)something didFinishLoadingItem:(id)item;
- (void)something:(id)something didFailWithError:(NSError *)error;
@end

@interface Something : NSObject
@property (nonatomic, weak) id <SomethingDelegate> delegate;
@end

@implementation Something {
  struct {
    unsigned int didFinishLoadingItem:1;
    unsigned int didFailWithError:1;
  } delegateRespondsTo;
}
@synthesize delegate;

- (void)setDelegate:(id <SomethingDelegate>)aDelegate {
  if (delegate != aDelegate) {
    delegate = aDelegate;

    delegateRespondsTo.didFinishLoadingItem = [delegate respondsToSelector:@selector(something:didFinishLoadingItem:)];
    delegateRespondsTo.didFailWithError = [delegate respondsToSelector:@selector(something:didFailWithError:)];
  }
}
@end

然后,在主体中,我们可以通过访问我们的delegateRespondsTo结构来检查我们的委托是否处理消息,而不是通过一次又一次地发送-respondsToSelector:

非正式代表

在协议存在之前,通常在NSObject上使用一个类别来声明委托可以实现的方法。 例如, CALayer仍然这样做:

@interface NSObject(CALayerDelegate)
- (void)displayLayer:(CALayer *)layer;
// ... other methods here
@end

这基本上告诉编译器,任何对象都可能实现displayLayer:

然后,您将使用与上述相同的-respondsToSelector:方法来调用此方法。 代表只需实现此方法并分配delegate属性,就是这样(没有声明符合协议)。 这种方法在Apple的库中很常见,但是新代码应该使用上面更现代的协议方法,因为这种方法会污染NSObject (这会使自动补全的用处变得不那么有用),并且使编译器很难警告您输入错误和类似的错误。


获得批准的答案非常棒,但如果您正在寻找1分钟的答案,请尝试以下操作:

MyClass.h文件应该看起来像这样(添加带有注释的委托行!)

#import <BlaClass/BlaClass.h>

@class MyClass;             //define class, so protocol can see MyClass
@protocol MyClassDelegate <NSObject>   //define delegate protocol
    - (void) myClassDelegateMethod: (MyClass *) sender;  //define delegate method to be implemented within another class
@end //end protocol

@interface MyClass : NSObject {
}
@property (nonatomic, weak) id <MyClassDelegate> delegate; //define MyClassDelegate as delegate

@end

MyClass.m文件应该看起来像这样

#import "MyClass.h"
@implementation MyClass 
@synthesize delegate; //synthesise  MyClassDelegate delegate

- (void) myMethodToDoStuff {
    [self.delegate myClassDelegateMethod:self]; //this will call the method implemented in your other class    
}

@end

要在另一个类中使用委托(本例中称为MyVC的UIViewController)MyVC.h:

#import "MyClass.h"
@interface MyVC:UIViewController <MyClassDelegate> { //make it a delegate for MyClassDelegate
}

MyVC.m:

myClass.delegate = self;          //set its delegate to self somewhere

实现委托方法

- (void) myClassDelegateMethod: (MyClass *) sender {
    NSLog(@"Delegates are great!");
}

当使用正式协议方法创建委托支持时,我发现可以通过添加如下内容来确保正确的类型检查(尽管是运行时,而不是编译时):

if (![delegate conformsToProtocol:@protocol(MyDelegate)]) {
    [NSException raise:@"MyDelegate Exception"
                format:@"Parameter does not conform to MyDelegate protocol at line %d", (int)__LINE__];
}

在你的委托访问器(setDelegate)代码中。 这有助于减少错误。

链接地址: http://www.djcxy.com/p/35143.html

上一篇: How do I create delegates in Objective

下一篇: changing constants for unit tests