利用KVC实现无需协议的委托模式

时间:2021-10-02 19:37:58

在《精通iOS开发》一书中看到的技巧。假设BIDTaskListController是一个列表,点击列表上的一项将会导航到BIDTaskDetailController,在BIDTaskDetailController当中修改值并保存后,将把修改后的值回传给BIDTaskListController并更新局部视图。

在BIDTaskListController类中有如下方法:

 - (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender {
UIViewController *destination = segue.destinationViewController;
if ([destination respondsToSelector:@selector(setDelegate:)]) {
[destination setValue:self forKey:@"delegate"];
}
if ([destination respondsToSelector:@selector(setSelection:)]) {
// prepare selection info
NSIndexPath *indexPath = [self.tableView indexPathForCell:sender];
id object = self.tasks[indexPath.row];
NSDictionary *selection = @{@"indexPath" : indexPath,
@"object" : object};
[destination setValue:selection forKey:@"selection"];
}
}

红字部分通过segue拿到了目标控制器,并且通过KVC的方式将自身设置为了目标的代理。

在BIDTaskDetailController当中有如下回调:

 [self.delegate setValue:editedSelection forKey:@"editedSelection"];

这句代码调用了BIDTaskListController当中editedSelection属性的setter方法,实现了对数据源的修改,并且在setter方法中刷新了视图:

 - (void)setEditedSelection:(NSDictionary *)dict {
if (![dict isEqual:self.editedSelection]) {
_editedSelection = dict;
NSIndexPath *indexPath = dict[@"indexPath"];
id newValue = dict[@"object"];
[self.tasks replaceObjectAtIndex:indexPath.row withObject:newValue];
[self.tableView reloadRowsAtIndexPaths:[NSArray arrayWithObject:indexPath]
withRowAnimation:UITableViewRowAnimationAutomatic];
}
}

如此,借助KVC以及回调数据源的setter方法可以彻底解除两个控制器之间的耦合,同时避免了定义协议,使代码更灵活。