我是Objective-c的新手,我仍然不清楚以下是否可能
编辑:
这是我使用和实施协议的方式:
@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”。我的理解是正确的还是合理的呢?
答案 0 :(得分:3)
没有协议不能从它们创建实例,如果它们可能只是类。协议只是定义任何类可以实现的一组方法的一种方式,这样您就不必使用继承来使用相同的方法来创建两个不同的类。对正在共享的类进行子类化会打扰父类和实现的接口,有时候在不共享任何实现的情况下共享类之间的接口是有意义的。
答案 1 :(得分:2)
协议只是一个接口 - 没有实现。因此即使你可以实例化一个协议(你不能),你仍然需要一些方法来提供协议接口的实现。
协议的重点在于它允许您指定仅接口,并且任何类都可以实现该协议。例如,iOS中的MapKit框架提供MKAnnotation
协议,该协议定义了地图注释对象的接口。您可以在任何您喜欢的课程中实施MKAnnotation
。如果您在某些类别中采用MKAnnotation
,那么您基本上承诺该类提供某些方法,即-coordinate
,-setCoordinate:
,-title
和{{1} }。在Objective-C这样的单继承语言中,这样做的能力很重要 - 这意味着您可以使任何类成为一个注释,您不必为某些提供的注释类创建子类
您的-subtitle
属性指向实现datePickerProtocol
的对象,而不是myProtocol
本身的实例(再次,这是没有意义的)。如果这真的是你想要的,那么你做得很好......你可以声明一个指向实现myProtocol
someProtocol
或id<someProtocol>
的对象的指针。
或者,如果要引用实际协议本身而不是实现协议的对象,则可以执行此操作。 Objective-C提供类似于NSObject<someProtocol>*
类型的Protocol
类型:就像任何给定的类是Class
类型的对象一样,任何给定的协议都是{{1}类型的对象}。您可以使用Class
指令获取特定协议的Protocol对象。 Objective-C Runtime Reference中有更多信息。