莫名其妙的崩溃
任何使用了UITextfiled的地方只要改变textfeild输入的值, 按HOME键退到后台,再打开APP 必崩溃;
但是只在iOS 8及以下系统的iPhone设备才会崩溃, iOS 9+系统就没事;
崩溃时没有任何log, 断点只能定位到这里:
曾经一度以为是苹果系统的bug,准备甩锅给苹果来着😆,还好宝宝比较敬业不找到bug睡不着;
于是各种尝试,各种调试,各种折腾 之后终于有点收获😁;
使用僵尸模式来调试:
输出了一行关键性代码:
这条log日志是说, release 方法发送给了一个已经销毁了的实例 "UIKeyboardLayoutStar";
这个鬼是个什么东西我也说不上来, 大概就是UITextfeild 在调用过程中会用到的一个控件吧;
但是, 要知道我们项目用的可是最先进的ARC技术, 并且我也没有自定义UITextfeild;
怎么会报这个错呢?真奇怪!
于是,好奇宝宝又是一通谷哥+度娘, 终于发现了问题所在, 推测是下图这个NSArray的分类里重写了load, 并使用方法交换 重定向objectAtIndex这个方法, 结果导致UITextfeild 调用过程中出现懵逼脸, 本该取出来的值没了,崩溃了;
又是一通谷哥+度娘之后, 宝宝找到两个解决方案:
1,不使用方法交换来重定向objectAtIndex , 而是给分类增加方法, 以新方法取代objectAtIndex;
2,将NSArray的分类改成MRC的, 不使用ARC自动管理内存, 并改写代码,适应MRC模式;
笔者选择使用第二种方案, 因为改动最小, 只需要改动这一个分类文件:
完美解决问题~话说适配真是操碎了心啊😢
小Tips:
控制台打印中文:
NSArray:
@implementation NSArray (Log)
- (NSString *)descriptionWithLocale:(id)locale
{
NSMutableString *strM = [NSMutableString stringWithString:@"(\n"];
[self enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
[strM appendFormat:@"\t%@,\n", obj];
}];
[strM appendString:@")"];
return strM;
}
@end
NSDictionary:
@implementation NSDictionary (Log)
- (NSString *)descriptionWithLocale:(id)locale
{
NSMutableString *strM = [NSMutableString stringWithString:@"{\n"];
[self enumerateKeysAndObjectsUsingBlock:^(id key, id obj, BOOL *stop) {
[strM appendFormat:@"\t%@ = %@;\n", key, obj];
}];
[strM appendString:@"}\n"];
return strM;
}
@end