一、问题及思路:
有人反馈键盘卡顿,但我手机上觉得键盘不卡呀,于是我觉得应该用工具来检测一下,而不是肉眼观察,主观判断。
二、解决过程:
我用Instruments里面的Timer Profile分析了一下程序,观察到键盘弹出的时候,执行了哪些方法,耗时多少。观察到有一个方法比较耗时:preshowImage方法,是一个判断最新一张图片的方法,由于遍历了相册中所有的照片,所以在手机中照片非常多的时候,是非常耗时的,而我的测试机中由于照片较少,所以没有那么卡。这是一个问题,代码应该写在输入框右边的加号点击事件中(类似微信),移除该方法了之后,点击输入框,弹出键盘好了很多。我又分析了一下代码,发现有一个方法比较耗时,是tableView reloadData。这句代码真的是既多余,又耗时(写代码的人太随意了)。去掉之后,又快了很多。继续分析代码,发现搜狗输入法这种第三方键盘,比系统键盘耗时,原因是:因为第三方键盘或者是在键盘加个toolbar会导致执行三次UIKeyboardWillShowNotification通知,原生键盘只执行一次,这下找到原因了,通过判断,可以做到第三方键盘也执行一次UIKeyboardWillShowNotification通知方法。
三、概括解决方法:
- 判断输入框,不调用遍历照片库方法,点击右边加号才调用;
- 第三方键盘调用了三次方法,通过判断,改成一次;
CGRect begin = [[[note userInfo] objectForKey:@"UIKeyboardFrameBeginUserInfoKey"] CGRectValue];
CGRect end = [[[note userInfo] objectForKey:@"UIKeyboardFrameEndUserInfoKey"] CGRectValue];
if(begin.size.height>0 && (begin.origin.y-end.origin.y>0)){
}
- 去掉无用代码reloadData;
- 遍历照片库,筛选掉视频,并且只遍历相机胶卷库。
- 排序照片库,取第一个,不遍历。这里其实还有点问题:相册默认是按日期倒序排列的,但是有没有可能改变相册的排序方式??后续我会再了解一下,再解决。因为这里影响不大,所以暂时先这样写了,性能上优化很多。官方文档这样描述ALAssetsGroup的:
An ALAssetsGroup is a ordered set of assets. The order of its elements is the order that the user sees in the Photos application
-
代码如下:
[assetsLibrary enumerateGroupsWithTypes:ALAssetsGroupSavedPhotos usingBlock:^(ALAssetsGroup *group, BOOL *stop) { [group setAssetsFilter:[ALAssetsFilter allPhotos]]; if (group) { [group enumerateAssetsWithOptions:NSEnumerationReverse usingBlock:^(ALAsset *result, NSUInteger index, BOOL *stop) { if (result) { NSDate *date = [result valueForProperty:ALAssetPropertyDate]; if ([date compare:lastDate] == NSOrderedDescending) { lastDate = date; lastPhoto = result; *stop = YES; } } }]; }
四、经验总结
这个问题最开始是这样的:
- 有的人反馈键盘卡,有的人不卡;
- 搜狗输入法比系统键盘卡;
解决之后,大概知道了原因:
- 因为有的人相册照片非常多,遍历的次数更多,更耗时。
- 搜狗输入法执行三次UIKeyboardWillShowNotification通知,导致键盘弹出的方法执行了三次,导致耗时;
以后遇到这样的问题,还是应该用工具来检测。尊重客观事实,而不是肉眼观察,主观判断。不能因为遇到问题的人少,就不去解决问题,经过分析,此处代码确实存在很多的问题,写的很不好。
后记:解决这个问题的时候,还遇到了Instruments的一些问题,后续专门写一篇文章来总结下Instruments的实际使用过程中遇到的问题及解决方法。