iOS11方法混淆导致tableView edit Cell异常

2018-07-03  本文已影响11人  coderPoo
一.先放问题现象
二.凭借祖传的开发经验猜测问题所在

首先想到的是不是iOS11的 tableView 新方法的问题,或者是开发中使用有问题

// Swipe actions
// These methods supersede -editActionsForRowAtIndexPath: if implemented
// return nil to get the default swipe actions
- (nullable UISwipeActionsConfiguration *)tableView:(UITableView *)tableView leadingSwipeActionsConfigurationForRowAtIndexPath:(NSIndexPath *)indexPath API_AVAILABLE(ios(11.0)) API_UNAVAILABLE(tvos);
- (nullable UISwipeActionsConfiguration *)tableView:(UITableView *)tableView trailingSwipeActionsConfigurationForRowAtIndexPath:(NSIndexPath *)indexPath API_AVAILABLE(ios(11.0)) API_UNAVAILABLE(tvos);

项目中的代码

- (UISwipeActionsConfiguration *)tableView:(UITableView *)tableView trailingSwipeActionsConfigurationForRowAtIndexPath:(NSIndexPath *)indexPath API_AVAILABLE(ios(11.0))
{
   if (indexPath.row >= self.chatData.count)
       return nil;
   
   GZIMChatModel *chatModel = self.chatData[indexPath.row];
   WS(weakSelf);
   UIContextualAction *deleteAction = [UIContextualAction contextualActionWithStyle:UIContextualActionStyleDestructive title:@"删除" handler:^(UIContextualAction * _Nonnull action, __kindof UIView * _Nonnull sourceView, void (^ _Nonnull completionHandler)(BOOL)) {
       [weakSelf p_deleteChat:chatModel completeHandler:completionHandler];
   }];
   UIContextualAction *topRowAction = [UIContextualAction contextualActionWithStyle:UIContextualActionStyleNormal title:(chatModel.chat.isTop?@"取消\n置顶":@"置顶") handler:^(UIContextualAction * _Nonnull action, __kindof UIView * _Nonnull sourceView, void (^ _Nonnull completionHandler)(BOOL)) {
       [weakSelf p_changeTopStatusWithChat:chatModel completeHandler:completionHandler];
   }];
   UIContextualAction *disturbRowAction = [UIContextualAction contextualActionWithStyle:UIContextualActionStyleNormal title:(chatModel.chat.unreadCount?@"标为\n已读":@"标为\n未读") handler:^(UIContextualAction * _Nonnull action, __kindof UIView * _Nonnull sourceView, void (^ _Nonnull completionHandler)(BOOL)) {
       [weakSelf p_changeDisturbStatusWithChat:chatModel completeHandler:completionHandler];
   }];
   disturbRowAction.backgroundColor = [UIColor lightGrayColor];
   
   UISwipeActionsConfiguration *config;
   if (chatModel.chat.chatType==GZIMChatType_Subscription) {
       config = [UISwipeActionsConfiguration configurationWithActions:@[deleteAction]];
   } else {
       config = [UISwipeActionsConfiguration configurationWithActions:@[deleteAction,topRowAction,disturbRowAction]];
   }
   config.performsFirstActionWithFullSwipe = NO;
   return config;
}

google查阅很多网上资料使用方式基本没有问题,而且网上也没有类似的问题出现

三、 分析项目多个版本以及最近业务逻辑

通过sourceTree代码回到上一个版本的tag位置,发现没有类似问题
基本能确定这是这一版本出现的问题
最后发现因为项目中最近加了 防止数组越界或者字典传nil 的代码混淆机制导致这个问题

@implementation NSMutableDictionary (NilSafe)

+ (void)load
{
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        Class class = NSClassFromString(@"__NSDictionaryM");
        [class exchangeInstanceMethod:@selector(setObject:forKey:) swizzleMethod:@selector(safeSetObject:forKey:)];
        //问题出现在这个上
        [class exchangeInstanceMethod:@selector(setObject:forKeyedSubscript:) swizzleMethod:@selector(safeSetObject:forKeyedSubscript:)];
        [class exchangeInstanceMethod:@selector(removeObjectForKey:) swizzleMethod:@selector(safeRemoveObjectForKey:)];
    });
}

- (void)safeSetObject:(id)anObject forKey:(id<NSCopying>)aKey {
    if (!aKey || !anObject) {
        XLOG_INFO(@"NSMutableDictionary setObjectForKey aKey or anObject is nil");
        return;
    }
    [self safeSetObject:anObject forKey:aKey];
}

- (void)safeSetObject:(id)obj forKeyedSubscript:(id<NSCopying>)key {
    if (!key || !obj) {
        XLOG_INFO(@"NSMutableDictionary setObjectForKeyedSubscript aKey is: %@ or anObject is %@",key,obj);
        return;
    }
    XLOG_INFO(@"NSMutableDictionary aKey is: %@ and anObject is:%@",key,obj);
    [self safeSetObject:obj forKeyedSubscript:key];
}

- (void)safeRemoveObjectForKey:(id)aKey
{
    if (!aKey) {
        XLOG_INFO(@"NSMutableDictionary removeObjectForKey aKey is nil");
        return;
    }
    return [self safeRemoveObjectForKey:aKey];
}

@end

左滑编辑时的 正常的日志

2018-07-03 18:25:19.121977+0800 GZIMClient[30512:2773379] [I][GZIM][NSObject+Swizzling.m, -[NSMutableDictionary(NilSafe) safeSetObject, 247][Info:NSMutableDictionary aKey is: <NSIndexPath: 0xc000000000000016> {length = 2, path = 0 - 0} and anObject is:<UISwipeOccurrence: 0x6040002f4200>

左滑编辑时的 异常的日志

2018-07-03 18:25:20.076750+0800 GZIMClient[30512:2773379] [I][GZIM][NSObject+Swizzling.m, -[NSMutableDictionary(NilSafe) safeSetObject, 244][Info:NSMutableDictionary setObjectForKeyedSubscript aKey is: <NSIndexPath: 0xc000000000800016> {length = 2, path = 0 - 4} or anObject is (null)

说明iOS11之后 新编辑方法用到了 下边这个方法

- (void)setObject:(nullable ObjectType)obj forKeyedSubscript:(KeyType <NSCopying>)key API_AVAILABLE(macos(10.8), ios(6.0), watchos(2.0), tvos(9.0));

而我们又代码混淆了这个方法,导致了问题的出现,应该是混淆导致 UISwipeOccurrence没有传入,不能正确获取,导致错位
解决方法: 只能先注销到这段方法混淆

四、 疑问

不明白为什么会出现这种问题,应该是内部原理

上一篇 下一篇

猜你喜欢

热点阅读