前言
开发做笔记是好习惯,总结分享是巩固记忆。
遇到问题,思考其背后的原因、原理。
AFNetworking
- 1、progress回调block,不在主线程;
- 2、iPhone4+iOS7,progress回调异常;
AFNetworking 处理请求是在后台线程,所以block回调的时候最好加一个dispatch。
我们默认把所有的网络请求在后台线程进行,但是回调的时候dispatch到主线程。
关闭键盘
在任意界面,dismiss键盘。(苹果的响应链设计)
[[UIApplication sharedApplication] sendAction:@selector(resignFirstResponder) to:nil from:nil forEvent:nil];
苹果的解释:
The object to receive the action message. If target is nil, the app sends the message to the first responder, from whence it progresses up the responder chain until it is handled.
类似AS3的冒泡查找响应者。
Provisioning Profile
iOS Team Provisioning Profile是第一次使用Xcode添加设备时,Xcode自动生成的,它包含了Xcode生成的一个Wildcard App ID(*,匹配所有应用程序),账户里面所有的Devices和所有Development Certificates,如下图所示。因此,team中的所有成员都可以使用这个iOS Team Provisioning Profile在team中的所有设备上调试所有的应用程序。并且当有新设备添加进来时,Xcode会更新这个文件。
开发者QA
苹果的安全认证体系:如何用证书、APPID、profile来验证开发者身份、IPA包正确性?
基本要求:保证设备的安全;验证开发者身份。
UILongpressGestureRecognizer
给view添加UILongpressGestureRecognizer,
触发长按之后,UILongpressGestureRecognizer的识别范围也会包含其superView;同时,触发长按之后,UIButton的高亮状态就会消失。
这是因为识别为长按之后就会将UIButton的touch cancel掉,UIButton不会给长按设置highlighted状态。
在begin后设置UIButton的selected为YES,移出UIButton后设置为NO,这样就可以实现长按点击的效果。
- (void)longPressAction:(UILongPressGestureRecognizer *)recognizer {
if (recognizer.state == UIGestureRecognizerStateBegan) {
bLongPress = YES;
[self sendAction];
_dodgersBtn.selected = YES;
}
else if (recognizer.state == UIGestureRecognizerStateEnded) {
bLongPress = NO;
_dodgersBtn.selected = NO;
}
}
[_dodgersBtn setBackgroundImage:normalDodgerImg forState:UIControlStateNormal];
[_dodgersBtn setBackgroundImage:focusDodgerImg forState:UIControlStateSelected];
[_dodgersBtn setBackgroundImage:focusDodgerImg forState:UIControlStateHighlighted];
//三个按钮状态,都得设置。
Xcode
问题1:Your build settings specify a provisioning profile with the UUID “”,
however, no such provisioning profile was found
通常来说是因为项目中引用了一个无效的PROVISIONING_PROFILE。按command+F,在这个文件中查找“PROVISIONING_PROFILE"。查找对应UUID的profile是否存在其他地方的引用。
问题2:Xcode 7.2 Organizer无限菊花、无限加载、卡顿如何解决?
1、证书相关:Preferences -> Accounts -> Account -> ViewDetails
2、文件过多:删除下列文件夹中无用文件
/Users/user/Library/Developer/Xcode/Products
/Users/loyinglin/Library/Developer/Xcode
问题3:nib but the view outlet was not set
nib文件 -> File's Owner -> command+4 >查看 Class 属性
检查Files's Owner的view是否有关联
PS:不关联,Xcode不知道是谁的类,此时无法进行合适的编译。(Xcode在编译的时候就会把xib处理成一个中间文件(xml=>nib),以便于在运行时加载)
问题4:A valid provisioning profile for this executable was not found
检查 Project 和 Target 的Code Signing Identity 与Provisioning Profile 设置是否一致;Provisioning Profile会告诉Xcode哪些证书是允许签名。(见上面的图)
问题5:debug时如何查看崩溃地址?
如果是异常引起的Crash,可以用下面的方法。
通用的解决方案是在命令行里打bt,会出堆栈。
Category
用Category来分离业务时,在Category使用了ActionSheet。如果对应的ViewController也实现了- (void)actionSheet:(UIActionSheet *)actionSheet didDismissWithButtonIndex:(NSInteger)buttonIndex
,逻辑就会存在问题。
category尽量不存在和原ViewController一致的方法。所以希望对category按照业务进行聚合,避免冲突。
RunLoop
RunLoop用来系统唤醒休眠线程,处理异步的事件,比如说url请求。
默认performSelectorOnMainThread:withObject:waitUntilDone:
是在默认的Runloop mode,如果runloop处于其他mode(比如说tracking mode),它会等待runloop切换回默认的mode。
-performSelectorOnMainThread:withObject:waitUntilDone:modes:
可以选择添加到的runloop mode。
dispatch_async(dispatch_get_main_queue(), ^{ })
会尽可能快的去执行block,而不管modes。
每个线程都会关联到不同的RunLoop,线程的Runloop可以工作在不同的modes。
RunLoop博客
总结
写于2018年7月,对文章进行一期整理,把知识点挖得更深入。