介绍
1.导入主头文件 #import <CoreLocation/CoreLocation.h>
2.地图和定位功能基于2个框架进行开发:
(1)Map Kit :用于地图展示
(2)CoreLocation:用于地理定位,有时只用定位,比如外卖,只有需要显示地图才用map kit
3.2个热门专业术语:
(1)LBS :Location Based Service 位置服务,又称定位服务
LBS的服务归纳为四类:定位(个人位置定位)、导航(路径导航)、查询(查询某个人或某个对象)、识别(识别某个人或对象)、事件检查(当出现特殊情况下向相关机构发送带求救或查询的个人位置信息)。
(2)SoLoMo :Social Local Mobile(索罗门) 移动社交
4.天朝的经纬度范围:纬度范围:N 3°51′ ~ N 53°33′ 经度范围:E 73°33′ ~ E 135°05′
1、如果定位方法不走原因:
(1)没有配置 plist 键值
(2)模拟器 bug
(3)没有使用 strong 的属性
2、逻辑结构
请求用户授权方法
注意:
- 一步代码,一步plist配置
- 请求授权iOS8以后才有,一定注意版本适配!
- 如果同时实现两个请求, 第一次运行会弹出第一个, 第二次运行会弹出第二个,大部分应用程序只需要使用用户使用期间授权即可
-
如果程序列表中出现了3行(即永不定位,使用应用期间,始终定位),说明实现了2种授权
- plist添加的用户提示信息可以不写值,表示空白提示,一般写上需要定位的原因,提高用户打开的几率
1. 代码:
if ([UIDevice currentDevice].systemVersion.floatValue >= 8.0) {
if ([self.locationManager respondsToSelector:@selector(requestWhenInUseAuthorization)]) {
// 永久授权
[self.locationManager requestAlwaysAuthorization];
//用户使用时授权,大部分的应用应该使用此种授权方式,当能看见程序时才能定位
[self.locationManager requestWhenInUseAuthorization];
}
}
2. plist文件配置
① 使用期间授权:添加该键,值是用户提示信息
NSLocationWhenInUseUsageDescription
② 永久始终授权:添加该键,值是用户提示信息
NSLocationAlwaysUsageDescription
③ iOS9.0新特性:临时获取后台定位权限(永久授权)
注意:该方式进入后台后会有提示
代码:iOS9 临时开启后台定位, allowsBackgroundLocationUpdates属性设置为YES
if ([UIDevice currentDevice].systemVersion.floatValue >= 9.0) {
self.locationManager.allowsBackgroundLocationUpdates = YES;
}
plist:需要配置Plist,不然会崩溃,是一个数组,值添加到一个数组的对象当中
键 :Required background modes
数组值:App registers for location updates
CLLocationManager 属性和方法
注意:
- 想要使用定位, 必须使用CLLocationManager(位置管理器)创建一个对象
- iOS8以后, 要定位, 必须使用位置管理器授权,并配置plist
-
使用期间授权:APP退到后台就不进行定位了
始终授权:后台也会进行定位,例如记录跑步,持续定位需要对电量做些优化(设置两个属性)
- 一般请求定位放在appDelegate中,在加载页面之前就定位好
下面两个属性设置后可以降低代理方法调用频率(默认大约一秒调用一次),以此省电,始终持续定位时候建议使用
属性:
- 距离筛选器,每隔多少米定位一次,单位:米,当用户发生一定位置的改变时, 再去调用代理方法, 以此实现省电
@property(assign, nonatomic) CLLocationDistance distanceFilter;
例子:每隔十米定位一次
self.locationManager.distanceFilter = 10;
- 定位精确度(越精确就越耗电)
@property(assign, nonatomic) CLLocationAccuracy desiredAccuracy;
降低精准度,实际上降低了与卫星之间的计算,以此节省电量
Iphone的定位方式:(1)GPS(2)wifi定位(3)移动基站定位(流量)
例子:设置定位精度误差一千米
self.locationManager.desiredAccuracy = kCLLocationAccuracyKilometer;
枚举值:iOS9以前默认kCLLocationAccuracyBest
kCLLocationAccuracyBestForNavigation 最精确的定位
kCLLocationAccuracyBest; 最好的
kCLLocationAccuracyNearestTenMeters; 十米误差
kCLLocationAccuracyHundredMeters; 一百米
kCLLocationAccuracyKilometer; 一千米
kCLLocationAccuracyThreeKilometers; 三千米
方法:
1.开始用户定位
- (void) startUpdatingLocation;
2.停止用户定位
- (void) stopUpdatingLocation;
代理方法
当调用了startUpdatingLocation方法后,就开始不断地调用该代理方法定位用户的位置,locations参数里面装着一组CLLocation对象,持续定位需要设置 distanceFilter 和 desiredAccuracy 属性
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations;
CLLocation 位置对象介绍
CLLocation用来表示某个位置的地理信息,比如经纬度、海拔等等,创建一个 CLLocation对象只需要两个参数:纬度和经度
属性:
(1)2D位置坐标,经纬度
@property(readonly, nonatomic) CLLocationCoordinate2D coordinate;
CLLocationDegrees latitude 纬度
CLLocationDegrees longitude 经度
(2)海拔
@property(readonly, nonatomic) CLLocationDistance altitude;
(3)速度(单位是m/s)
@property(readonly, nonatomic) CLLocationSpeed speed;
(4)水平 垂直精准度
@property(readonly, nonatomic) CLLocationAccuracy horizontalAccuracy;
@property(readonly, nonatomic) CLLocationAccuracy verticalAccuracy;
(5)航向(取值范围是0.0° ~ 359.9°,0.0°代表真北方向)
@property(readonly, nonatomic) CLLocationDirection course;
(6)时间戳,什么时间进行的定位
@property(readonly, nonatomic, copy) NSDate *timestamp;
方法:
(1)创建一个 CLLocation对象只需要两个参数:纬度和经度
- (instancetype)initWithLatitude:(CLLocationDegrees)latitude
longitude:(CLLocationDegrees)longitude;
(2)计算2个位置之间的距离,比较的是直线距离,单位是米,除以1000可以换算成千米
- (CLLocationDistance)distanceFromLocation:(constCLLocation *)location;
例子:计算北京和西安的位置直线距离
CLLocation *location1 = [[CLLocation alloc] initWithLatitude:40.06 longitude:116.39];
CLLocation *location2 = [[CLLocation alloc] initWithLatitude:34.27 longitude:108.93];
CGFloat distance = [location2 distanceFromLocation:location1];
NSLog(@"distance: %f",distance / 1000);
例子:定位功能实现
1、 创建位置管理器
self.locationManager = [CLLocationManager new];
2、 请求用户授权(iOS8以后才有) 同时配置 plist 列表,注意:必须使用版本判断,建议结合使用
-----------------------------
有两种授权方式,还有一种始终 requestAlwaysAuthorization
if([self.locationManagerrespondsToSelector:@selector(requestWhenInUseAuthorization)]) {
[self.locationManager requestWhenInUseAuthorization];
}
3、 设置 <CLLocationManagerDelegate>代理, 来获取用户位置数据
self.locationManager.delegate = self;
4、调用开始定位方法
[self.locationManager startUpdatingLocation];
5、实现代理方法拿到数据,当用户更新位置的时候调用此方法,频繁调用, 非常耗电
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray<CLLocation *> *)locations{
//CLLocation 位置对象 --> 经纬度
//CLLocationCoordinate2D coordinate 经纬度
//获取最后一次位置信息
CLLocation *location = locations.lastObject;
//输出纬度和经度
NSLog(@"latitude: %f,longitude: %f",location.coordinate.latitude, location.coordinate.longitude);
// 停止定位
[self.locationManager stopUpdatingLocation];
}
三、地理编码的实现
- 正地理编码:将地名转换成经纬度的过程
步骤:
(1)创建一个GLGeocoder对象
(2)实现地理编码方法
(3)遍历数组,获取数据(可能返回多个相同地名),如果对象大于1,应该给用户一个列表选择
- 反地理编码:将经纬度转换成地名的过程
步骤:
(1)创建一个GLGeocoder对象
(2)创建一个CLLoction对象(经纬度)
(3)实现反地理编码方法
(4)遍历数组,获取数组
CLGeocoder地理编码对象
一个属性:
@property (nonatomic, readonly, getter=isGeocoding) BOOL geocoding;
方法:
1、三个正地理编码方法
(1)
- (void)geocodeAddressDictionary:(NSDictionary *)addressDictionary completionHandler:(CLGeocodeCompletionHandler)completionHandler;
(2)最简单的,填入要搜索的位置,回调出搜索信息
- (void)geocodeAddressString:(NSString *)addressString completionHandler:(CLGeocodeCompletionHandler)completionHandler;
(3)
- (void)geocodeAddressString:(NSString *)addressString inRegion:(nullable CLRegion *)region completionHandler:(CLGeocodeCompletionHandler)completionHandler;
2、一个反地理编码方法
- (void)reverseGeocodeLocation:(CLLocation *)location completionHandler:(CLGeocodeCompletionHandler)completionHandler;
3、取消编码
- (void)cancelGeocode;
回调的block参数CLGeocodeCompletionHandler
typedef void (^CLGeocodeCompletionHandler)(NSArray< CLPlacemark *> * __nullable placemarks, NSError * __nullable error);
block参数:
- placemarks:CLPlacemark 地标对象,里面封装各种详细地址信息,如街道名,城市名,国家名等
- error :当编码出错时有值(比如编码不出具体的信息)
CLPlacemark 地标对象:
- (instancetype)initWithPlacemark:(CLPlacemark *) placemark;
1、主要获取的属性,位置对象,能获取到经纬度信息
@property (nonatomic, readonly, copy, nullable) CLLocation *location;
@property (nonatomic, readonly, copy, nullable) CLRegion *region;
@property (nonatomic, readonly, copy, nullable) NSTimeZone *timeZone ( 9_0))
@property (nonatomic, readonly, copy, nullable) NSDictionary *addressDictionary;
详细地址名(包括等到门牌等)
@property (nonatomic, readonly, copy, nullable) NSString *name;
街道名
@property (nonatomic, readonly, copy, nullable) NSString *thoroughfare;
子街道名
@property (nonatomic, readonly, copy, nullable) NSString *subThoroughfare;
城市
@property (nonatomic, readonly, copy, nullable) NSString *locality;
子城市
@property (nonatomic, readonly, copy, nullable) NSString *subLocality;
行政区域
@property (nonatomic, readonly, copy, nullable) NSString *administrativeArea;
子行政区域
@property (nonatomic, readonly, copy, nullable) NSString *subAdministrativeArea;
@property (nonatomic, readonly, copy, nullable) NSString *postalCode;
@property (nonatomic, readonly, copy, nullable) NSString *ISOcountryCode;
国家
@property (nonatomic, readonly, copy, nullable) NSString *country;
@property (nonatomic, readonly, copy, nullable) NSString *inlandWater;
@property (nonatomic, readonly, copy, nullable) NSString *ocean;
@property (nonatomic, readonly, copy, nullable) NSArray<NSString *> *areasOfInterest;
正地理编码例子
//1. 创建 Geocoder
CLGeocoder *geocoder = [CLGeocoder new];
//2. 调用方法
[geocoder geocodeAddressString:self.addressTF.text completionHandler:^(NSArray<CLPlacemark *> * _Nullable placemarks, NSError * _Nullable error) {
//3.1 防错处理
if (placemarks.count == 0 || error) {
NSLog(@"没有数据或数据解析出错");
return;
}
//3.2 遍历地标数组placemarks,地理编码容易出现多个地标,应该给用户一个列表去选择
for (CLPlacemark *pm in placemarks) {
//3.3 设置纬度
self.latitudeLabel.text = [NSString stringWithFormat:@"%f",pm.location.coordinate.latitude];
//3.4 设置经度
self.longitudeLabel.text = [NSString stringWithFormat:@"%f",pm.location.coordinate.longitude];
//3.5 设置地址
self.detailAddressLabel.text = pm.name;
}
}];
反地理编码例子
//1. 创建 Geocoder 对象
CLGeocoder *geocoder = [CLGeocoder new];
//2. 创建 CLLocation对象,输入经纬度信息
CLLocation *location = [[CLLocation alloc] initWithLatitude:[self.latitudeTF.text floatValue] longitude:[self.longitudeTF.text floatValue]];
//3. 调用反地理编码方法
[geocoder reverseGeocodeLocation:location completionHandler:^(NSArray<CLPlacemark *> * _Nullable placemarks, NSError * _Nullable error) {
//3.1 防错处理
if (placemarks.count == 0 || error) {
NSLog(@"没有数据或数据解析出错");
return;
}
//3.2 遍历数据
for (CLPlacemark *pm in placemarks) {
//3.3. 获取城市信息, 如果有城市信息就显示, 否则可以显示行政区域
if (pm.locality) {
//locality : 城市
self.cityLabel.text = pm.locality;
} else {
//administrativeArea : 行政区域
self.cityLabel.text = pm.administrativeArea;
}
}
}];