如果,你的老项目在iOS8下运行,打开就闪退(iOS8之前没问题),那么“恭喜你”,你中招了,比如下面我遇到的,是因为旧版本的高德地图引用了 iOS8 里面不能用的api,如果你也需要类似的问题,那么是时候升级需要升级的第三方库了。
2014-09-28 14:32:25.576 WoZaiXianChang[4505:140022] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[UIDevice asUniqueDeviceIdentifier]: unrecognized selector sent to instance 0x7c020080'
之前版本的SDk是这样启动系统定位的
// 判断定位操作是否被允许
if([CLLocationManager locationServicesEnabled]) {
locationManager = [[CLLocationManager alloc] init];
locationManager.delegate = self;
[locationManager startUpdatingLocation];
}else {
//提示用户无法进行定位操作
}
如果在iOS8下用这样的方式,你会发现无法定位,那是因为iOS8下添加了新的方法
/表示使用应用程序期间 开启定位
- (void)requestWhenInUseAuthorization
//表示始终 开启定位
- (void)requestAlwaysAuthorization
两者区别在于,iOS7 开始,有更强大的后台运行功能,如果 用 requestAlwaysAuthorization 方法,则表示后台运行时也会用到定位
iOS8 下使用系统定位如下:
// 判断定位操作是否被允许
if([CLLocationManager locationServicesEnabled]) {
locationManager = [[CLLocationManager alloc] init];
locationManager.delegate = self;
//兼容iOS8定位
SEL requestSelector = NSSelectorFromString(@"requestWhenInUseAuthorization");
if ([CLLocationManager authorizationStatus] == kCLAuthorizationStatusNotDetermined &&
[locationManager respondsToSelector:requestSelector]) {
[locationManager requestWhenInUseAuthorization];
} else {
[locationManager startUpdatingLocation];
}
return YES;
}else {
//提示用户无法进行定位操作
}
return NO;
同时还需要添加新的方法,其他的都一样
- (void)locationManager:(CLLocationManager *)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status {
if (status == kCLAuthorizationStatusAuthorizedWhenInUse) {
[locationManager startUpdatingLocation];
} else if (status == kCLAuthorizationStatusAuthorized) {
// iOS 7 will redundantly call this line.
[locationManager startUpdatingLocation];
} else if (status > kCLAuthorizationStatusNotDetermined) {
//...
[locationManager startUpdatingLocation];
}
}
除了这些,你还需要在 info.plist 里面添加新的键值,否则 也是无法定位的
//表示使用应用程序期间 开启定位
- (void)requestWhenInUseAuthorization 对应 NSLocationWhenInUseUsageDescription key
//表示始终 开启定位
- (void)requestAlwaysAuthorization 对应 NSLocationAlwaysUsageDescription key
其中,NSLocationWhenInUseUsageDescription(或者NSLocationAlwaysUsageDescription) 对应的文字会在第一次请求用户同意定位的时候出现,还有 设置 > 隐私 > 定位 > your app 里面也会看到,比如下面就是开启app时出现的
这个不用多说,直接看代码就明白了,有一点需要注意的是,蓝色部分必须要加,不然即便能取的token值,app 接受到的推送也是无声的。
//注册消息通知
if (IOS8After) {
[[UIApplication sharedApplication] registerForRemoteNotifications];
[[UIApplication sharedApplication] registerUserNotificationSettings:[UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge | UIUserNotificationTypeSound | UIUserNotificationTypeAlert categories:nil]];
}
else {
[[UIApplication sharedApplication] registerForRemoteNotificationTypes:(UIRemoteNotificationTypeAlert | UIRemoteNotificationTypeSound | UIRemoteNotificationTypeBadge)];
}
如果你像这样取cell 的row 的话,那你又要加个判断方法了,在iOS8下cell的层级又改了,基本上每升级一个版本,苹果都会对cell的结构进行调整,在此建议不要用这样的方式取cell 的row,而是用属性的方式保存 indexPath
NSUInteger curRow = 0;
if ([[MetaData getOSVersion] integerValue] == 7)
{
curRow = [(UITableView )[[self superview] superview] indexPathForCell:self].row;
}
else
{
curRow = [(UITableView )[self superview] indexPathForCell:self].row;
}
在iOS8里面,官方提供了新的类UIAlertController
来替换UIActionSheet and UIAlertView。
示例代码如下:
UIAlertController* alert = [UIAlertController alertControllerWithTitle:@”My Alert”
message:@”This is an alert.”
preferredStyle:UIAlertControllerStyleAlert];
UIAlertAction defaultAction = [UIAlertAction actionWithTitle:@”OK” style:UIAlertActionStyleDefault
handler:^(UIAlertAction action) {}];
[self presentViewController:alert animated:YES completion:nil];
至于为什么为加这个类,本人猜测是和iOS8新加的size classes有关,目的是统一屏幕在各个尺寸各个方向上的显示。如果你在iOS 8 里面使用UIActionSheet and UIAlertView
可能会出现一些很奇怪的问题,建议在iOS 8 里面使用UIAlertController
,iOS 8 之前使用UIActionSheet and UIAlertView
转载: