CoreLocation + MapKit
CoreLocation
CoreLocation概述
- CoreLocation:用于地理定位,地理编码,区域监听(注重功能实现 )
- 使用CoreLocation之前要先导入<CoreLocation/CoreLocation.h>框架
- <CoreLocation/CoreLocation.h>框架中所有的数据类型是由CL开头的,使用CLLocationManager对象来做用户定位的
- CLLocationManager
方法 |
功能 |
desiredAccuracy |
定位的精度, 定位的越精确就会越耗电 |
distanceFilter |
每隔多少米定位一次 |
locationServicesEnabled |
判断当前应用的定位功能是否可用 |
startUpdatingLocation |
开始更新用户的位置 |
stopUpdatingLocation |
停止更新用户的位置 |
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations |
设置开启更新用户位置,就会来到代理方法,设置distanceFilter属性之后只有触发距离约束才会再次走这个方法,如果不设置会一直走 |
- (void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error |
获取位置信息失败 |
authorizationStatus |
当前应用程序的授权状态(枚举,两个常用)kCLAuthorizationStatusAuthorizedWhenInUse和kCLAuthorizationStatusAuthorizedAlways(只有授权了才可以定位,前者是前台定位,后者是推入后台也可以定位) |
requestWhenInUseAuthorization |
请求前台授权 |
requestAlwaysAuthorization |
请求前后台授权 |
注意:当请求授权之后,一定要在info.plist文件中加一条前台:NSLocationWhenInUseUsageDescription(后面写显示的东西)后台:NSLocationAlwaysUsageDescription(8.0之后)
- CLLocation(用于表示某个位置的地理信息)
方法 |
功能 |
coordinate |
经纬度 |
altitude |
海拔 |
course |
路线,航向(取值范围是0.0° ~ 359.9°,0.0°代表真北方向) |
speed |
移动速度(单位是m/s) |
#import "ViewController.h"
//定位框架
#import <CoreLocation/CoreLocation.h>
@interface ViewController ()<CLLocationManagerDelegate>
/**
* 定位管理者
*/
@property (nonatomic,strong)CLLocationManager *mgr;
/**
* 编码反编码
*/
@property (nonatomic,strong)CLGeocoder *geo;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
//创建定位管理者
self.mgr = [[CLLocationManager alloc] init];
//定位的精度
self.mgr.desiredAccuracy = 100;
//最小更新距离
self.mgr.distanceFilter = 100;
//开启定位
[self.mgr startUpdatingLocation];
//设置代理
self.mgr.delegate = self;
//当前手机的版本判断>=8.0;
if ([[[UIDevice currentDevice] systemVersion] integerValue] >= 8.0) {
//手机是否打开定位
if (![CLLocationManager locationServicesEnabled])
{
return;
}else
{
//如果用户拒绝
//弹框(设置->隐私->...)让客户去自己打开
}
//是否授权当前应用程序定位
if ([CLLocationManager authorizationStatus] !=kCLAuthorizationStatusAuthorizedWhenInUse) {
//如果不是这种授权状态就去请求授权状态(简单定位)(在info.plist添加相应的key值)
[self.mgr requestWhenInUseAuthorization];//ios8之后的方法
}
}
}
//位置信息
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray<CLLocation *> *)locations
{
NSLog(@"%f, %f",locations.lastObject.coordinate.latitude,locations.lastObject.coordinate.longitude);
//停止定位
[self.mgr stopUpdatingLocation];
}
//获取位置信息失败
- (void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error
{
NSLog(@"%@",error);
}
//懒加载
-(CLGeocoder *)geo
{
if (!_geo) {
_geo = [CLGeocoder new];
}
return _geo;
}
编码和反编码
- CLGeocoder: 可以完成地理编码和反编码,编码:根据给定的地名,获得具体的位置信息(经纬度等等),反编码:根据给定的经纬度获得具体的位置信息
方法 |
功能 |
- (void)geocodeAddressString:(NSString *)addressString completionHandler:(CLGeocodeCompletionHandler)completionHandler |
编码 |
- (void)reverseGeocodeLocation:(CLLocation *)location completionHandler:(CLGeocodeCompletionHandler)completionHandler |
反编码 |
- 编码和反编码返回的都是CLPlacemark对象的
- CLPlacemark:封装详细的地理位置信息
方法 |
功能 |
location |
地理位置 |
region |
区域 |
addressDictionary |
详细的地址信息 |
name |
地址名称 |
locality |
城市 |
MapKit
MapKit的基本使用
- 首先导入第三方框架<MapKit/MapKit.h>
- MapKit由于注重的是显示效果,所以所有展示的功能集成到了MKMapView这个view上,也就是一定要有一个MKMapView的对象
- 如果授权成功了就不必再授权了
- mapKit
方法 |
功能 |
showsUserLocation |
展示用户的位置 |
mapType |
地图的类型 |
rotateEnabled |
地图是否可以旋转 |
userTrackingMode |
追踪模式 |
- (void)mapView:(MKMapView *)mapView didUpdateUserLocation:(MKUserLocation *)userLocation |
拿到用户当前的位置信息(MKMapViewDelegate的方法) |
- (MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id<MKAnnotation>)annotation |
自定义大头针需要用到的方法(MKMapViewDelegate的方法) |
- (void)mapView:(MKMapView *)mapView regionWillChangeAnimated:(BOOL)animated |
监听地图的范围将要发生变化(MKMapViewDelegate的方法) |
- (void)mapView:(MKMapView *)mapView regionDidChangeAnimated:(BOOL)animated |
监听地图已经发生变化(MKMapViewDelegate的方法) |
0:10 |
1:10 |
//拿到用户位置的代理,并绘制一个当前位置的大头针
- (void)mapView:(MKMapView *)mapView didUpdateUserLocation:(MKUserLocation *)userLocation
{
//保证只会创建一个大头针(当用户的位置发生变化是,就会走这个方法,所以定位自身位置的大头针可能会被重复创建)
for (id annotation in mapView.annotations) {
if ([annotation isKindOfClass:[Annimation class]]) {
[mapView removeAnnotation:annotation];
}
}
//初始化大头针(在这里初始化的是定位自己当前位置的大头针,因为这个方法可以拿到当前的位置,如果需求是绘制其他位置的大头针可以不在这个方法中写)
Annimation *anni = [[Annimation alloc] init];
anni.coordinate = userLocation.coordinate;
anni.title = userLocation.title;
anni.subtitle = userLocation.subtitle;
anni.icon = @"123";
//添加大头针
[self.mapView addAnnotation:anni];
MKCoordinateRegion region = MKCoordinateRegionMake(userLocation.coordinate, MKCoordinateSpanMake(1, 1));
//设置地图展示的范围
[self.mapView setRegion:region animated:YES];
}
//自定义大头针需要的方法(只要地图上的大头针都会走这个方法,如果这个方法返回空那么就会走系统的大头针MKAnnotation)
//这里的原理是:给你一个id大头针模型(id<MKAnnotation>),需要绘制一个大头针(MKAnnotationView),这个大头针模型相当于是一个数据源,为我们的大头针提供显示的数据的
- (MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id<MKAnnotation>)annotation
{
//如果不是我们自定义的大头针,就返回系统默认的
if (![annotation isKindOfClass:[Annimation class]]) return nil;
//自定义的大头针(我们自定义的其实是大头针模型Annimation,跟cell的创建原理一样的,大头针是根据大头针模型创建的,而这里用到了cell的重用机制)
MKAnnotationView *view = [mapView dequeueReusableAnnotationViewWithIdentifier:@"1"];
if (!view) {
view = [[MKAnnotationView alloc] initWithAnnotation:annotation reuseIdentifier:@"1"];
}
//这里以下才是大头针真正的自定义
//赋值(其实上面初始化的时候已经赋值过了,如果不重新赋值,由于重用机制,就会导致数据错乱)
view.annotation = annotation;
//修改大头针的图片
view.image = [UIImage imageNamed:((Annimation *)annotation).icon];
//显示大头针的头视图
view.canShowCallout = YES;
return view;
}