2

我是objective-c的新手,我仍然不清楚以下是可能的

  1. 是否可以实例化协议?
  2. 如果可以实例化一个协议,我该如何实现所需的方法?

编辑:

这就是我使用和实现协议的方式:

@protocol myProtocol <NSObject>

@required
    - (void)doneAction:(NSDate *)dateSelected;
@end

@interface datePickerView : UIView

@property (strong, nonatomic) UIDatePicker *datePicker;
@property (strong, nonatomic) UIButton *doneButton;
@property (strong, nonatomic) NSObject<myProtocol> *datePickerProtocol;

- (id)initWithDetails:(UIDatePicker *)datePick doneBtn:(UIButton *)doneBtn;
- (void)show;
@end

这是“datePickerView”的实现

@synthesize datePicker = _datePicker;
@synthesize doneButton = _doneButton;
@synthesize datePickerProtocol = _datePickerProtocol;

- (id)initWithDetails:(UIDatePicker *)datePick doneBtn:(UIButton *)doneBtn
{
    _datePicker = datePick;
    _doneButton = doneBtn;
    [_doneButton addTarget:self action:@selector(doneButtonClicked) forControlEvents:UIControlEventTouchUpInside];

    [self addSubview:_datePicker];
    [self addSubview:_doneButton];
}

- (void)doneButtonClicked
{
    /** Code to hide the view goes here **/
    if (_datePickerProtocol != nil)
        [_datePickerProtocol doneAction:_datePicker.date];
}

这是使用“datePickerView”的类

/** .h file **/
@interface myController : UITableViewController <myProtocol>
{
}

@property (strong, nonatomic) IBOutlet UIDatePicker *datePicker;
@property (strong, nonatomic) IBOutlet UIButton *datePickDone;
/** More definitions goes here **/

/** .m file **/
@synthesize datePicker = _datePicker;
@synthesize datePickDone = _datePickDone;
/** More code goes here **/

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    UITableViewCell *selectedCell = [self.tableView cellForRowAtIndexPath:indexPath];
    if ((![selectedCell.reuseIdentifier isEqualToString:@"dateAppliedCellIdentifier"]) &&
        (![selectedCell.reuseIdentifier isEqualToString:@"closingDateCellIdentifier"]))
        return;

    if (_datePicker.superview == nil)
    {
        datePickerView = [[datePickerView alloc] initWithDetails:_datePicker     doneBtn:_datePickDone];

        [self.view.window addSubview: datePickerView];
    }
    /**
        THIS IS WHERE I WANT TO PASS AN INSTANCE OF THE PROTOCOL (BUT AS ADVISED,
        IT IS NOT POSSIBLE) DEPENDING ON WHICH CELL WAS TAPPED. THE REASON I WANT
        PASS AN INSTANCE OF THE PROTOCOL INSTEAD OF "SELF" IS BECAUSE I WANT
        TO AVOID DOING MULTIPLE "IF" STATEMENTS INSIDE
        "(void)doneAction:(NSDate *)dateSelected" METHOD. I WOULD LIKE TO DO THE IFs
        HERE AND PASS THE CORRECT INSTANCE OF THE PROTOCOL DEPENDING ON WHICH CELL
        WAS TAPPED. EACH INSTANCE OF THE PROTOCOLS HAVE DIFFERENT IMPLEMENTATIONS OF
        "doneAction".
    **/
    [datePickerView setDatePickerProtocol:self];
    [datePickerView show];
}

//implementation of the protocol method
- (void)doneAction:(NSDate *)dateSelected
{
    //IF STATEMENTS GOES HERE DEPENDING ON WHICH CELL WAS TAPPED
    NSLog(@"Done action is called!");
}

请参阅我对代码的评论,在此行上方 [datePickerView setDatePickerProtocol:self]。由于我无法实例化协议并分配给变量,除了从控制器实现“myProtocol”并在“doneAction”方法中执行多个 IF 语句之外,是否有替代或更好的方法来实现我的目标?

根据答案,我似乎需要定义实现“myProtocol”的类并创建这些类的实例并传递给“datePickerView setDatePickerProtocol”。我的理解正确还是这样做合理?

4

2 回答 2

3

没有协议不能从它们创建实例,如果可以的话,它们只是类。协议只是定义一组任何类都可以实现的方法的一种方式,这样您就不必使用继承来让两个不同的类具有相同的方法。子类化您共享的类会打扰父类的接口和实现,有时在类之间共享接口而不共享任何实现是有意义的。

于 2012-10-21T23:23:56.227 回答
2

协议只是一个接口——没有实现。所以即使你可以实例化一个协议(你不能),你仍然需要一些方法来提供协议接口的实现。

协议的全部意义在于它只允许您指定接口,并且任何类都可以实现该协议。例如,iOS 中的 MapKit 框架提供了MKAnnotation协议,该协议定义了地图注释对象的接口。你可以MKAnnotation在任何你喜欢的类中实现。如果你在你MKAnnotation的某个类中采用,你基本上是在保证该类提供某些方法,即-coordinate-setCoordinate:-title-subtitle。做到这一点的能力在像 Objective-C 这样的单继承语言中很重要——这意味着你可以创建任何你喜欢的类作为注解,你不必对某些提供的注解类进行子类化。

您的datePickerProtocol属性指向一个实现的对象myProtocol,而不是它自身的一个实例myProtocol(同样,这没有意义)。如果这真的是你想要的,那么你做得很好......你可以声明一个指向实现someProtocol为的对象的指针id<someProtocol>or NSObject<someProtocol>*

或者,如果您想引用实际协议本身而不是实现该协议的对象,您可以这样做。Objective-C 提供了与Protocoltype 类似的Class类型:就像任何给定的类都是 type 的对象一样Class,任何给定的协议都是 type 的对象Protocol。您可以使用该@protocol指令来获取特定协议的协议对象。Objective-C 运行时参考中有更多信息。

于 2012-10-21T23:41:39.413 回答