平时在写项目的时候可能会遇到需要使用定位服务的地方,比如说获取位置和导航等。因此这里我会使用OC自带的库以及苹果系统的地图来获取定位以及显示在地图上。
开始前的设置
在获取定位前,需要在项目文件的info中添加两个关键字,用于向用户请求定位服务。在请求定位服务的弹窗中会显示我们添加的字段
<key>Privacy - Location When In Use Usage Description
<value>使用程序的时候获取本机位置<key>Privacy - Location Always Usage Description
<value>总是获取本机位置
运行效果:
获取当前位置的经纬度
- 在写代码前需要添加对应的库以及设置相关的属性:
#import <UIKit/UIKit.h>
// 关于定位以及地图的库
#import <MapKit/MKMapView.h>
#import <MapKit/MapKit.h>
#import <CoreLocation/CoreLocation.h>@interface ViewController : UIViewController<CLLocationManagerDelegate, MKMapViewDelegate>//设置一个定位管理者的属性
@property (nonatomic, strong) CLLocationManager *locationManager;
//存储推算出当前的地理位置信息,这个属性用于获取当前位置信息推算出的信息,比如有了经纬度而推算出的省、市、区等
@property (nonatomic, strong) CLGeocoder *geoCoder;//定义属性获取存储到的位置信息
@property (nonatomic, retain) CLLocation *myLocation;//MKMapView是iOS中MapKit框架中的一个类,用于显示地图,并提供与地图相关的交互功能
@property (nonatomic, strong) MKMapView *mapView;@end
- 开始定位:
//开始定位
- (void)dingWei {self.locationManager = [[CLLocationManager alloc] init];//desiredAccuracy用于指定定位服务精度, kCLLocationAccuracyBest表示最高精度,这个设置的优势是提供了最准确的位置信息,但代价是更高的能耗。其他定位精度设置,例如 kCLLocationAccuracyNearestTenMeters 或 kCLLocationAccuracyHundredMeters,它们提供了较低的精度但更节能。self.locationManager.desiredAccuracy = kCLLocationAccuracyBest;//distanceFilter 属性用于设置设备移动的距离,当设备移动超过这个距离时,会触发位置更新。distanceFilter 被设置为 10.0 米。这意味着只有当设备移动超过 10.0 米时,才会触发新的位置更新。self.locationManager.distanceFilter = 10.0f;//检查设备的系统版本是否可以进行定位if ([CLLocationManager locationServicesEnabled]) {self.locationManager.delegate = self;//如果是,则调用 requestWhenInUseAuthorization 方法请求用户在应用处于前台时获取位置信息的授权。dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{[self.locationManager requestWhenInUseAuthorization];});self.geoCoder = [[CLGeocoder alloc] init];} else {NSLog(@"ERROR");}
}
CLLocationManager其相关的属性:
desiredAccuracy位置的精度属性,取值有如下几种:
kCLLocationAccuracyBest:精确度最佳
kCLLocationAccuracynearestTenMeters:精确度10m以内
kCLLocationAccuracyHundredMeters:精确度100m以内
kCLLocationAccuracyKilometer:精确度1000m以内
kCLLocationAccuracyThree:精确度3000m以内
- 在代理方法中获取需要的位置信息
//在代理方法中获取需要的位置信息
//下面的方法中,locations是一个数组类型,其最后一个元素就是我们的经纬度坐标,类型为CLLocation,如果想将它设置为属性,修饰符一定要是retain
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray<CLLocation *> *)locations {NSLog(@"%lu", locations.count);self.myLocation = locations.lastObject;NSLog(@"经度: %f, 纬度: %f", self.myLocation.coordinate.longitude, self.myLocation.coordinate.latitude);// 获取到位置后再进行反地理编码[self reverseGeocodeLocation:self.myLocation];
}- (void)reverseGeocodeLocation:(CLLocation *)location {[self.geoCoder reverseGeocodeLocation:location completionHandler:^(NSArray<CLPlacemark *> * _Nullable placemarks, NSError * _Nullable error) {if (placemarks.count > 0) {CLPlacemark *placemark = [placemarks objectAtIndex:0];NSLog(@"%@", placemark.name);NSString *city = placemark.locality;if (!city) {city = placemark.administrativeArea;}NSLog(@"位置名:%@", placemark.name);NSLog(@"街道:%@", placemark.thoroughfare);NSLog(@"子街道:%@", placemark.subThoroughfare);NSLog(@"市:%@", placemark.locality);NSLog(@"区:%@", placemark.subLocality);NSLog(@"国家:%@", placemark.country);// 在这里调用显示地图的方法[self showLocationOnMapWithLatitude:location.coordinate.latitude longitude:location.coordinate.longitude];}}];
}
- 改变定位权限
// 在此回调中处理定位权限的改变
- (void)locationManagerDidChangeAuthorization:(CLLocationManager *)manager {CLAuthorizationStatus status = manager.authorizationStatus;dispatch_async(dispatch_get_main_queue(), ^{if (status == kCLAuthorizationStatusAuthorizedWhenInUse || status == kCLAuthorizationStatusAuthorizedAlways) {// 在这里执行需要权限的操作,例如启动定位服务//设置允许在应用在后台运行时继续获取位置更新self.locationManager.allowsBackgroundLocationUpdates = NO;//开始获取设备的当前位置信息[self.locationManager startUpdatingLocation];self.geoCoder = [[CLGeocoder alloc] init];[self jiSuanDistance];self.mapView = [[MKMapView alloc] initWithFrame:self.view.bounds];self.mapView.delegate = self;[self.view addSubview:self.mapView];} else {NSLog(@"ERROR");}});
}
- 获取定位出错时调用如下方法:
//获取位置出错的时候调用下面的协议方法
- (void) locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error {if (error) {NSLog(@"ERROR");}
}
运行结果:
测算两个经纬度之间的距离
//测算两个经纬度坐标之间的距离
- (void) jiSuanDistance {CLLocation *before = [[CLLocation alloc] initWithLatitude:11.111 longitude:222.222];CLLocationDistance meters = [self.myLocation distanceFromLocation:before];NSLog(@"相距: %f米", meters);
}
运行结果:
将位置定位在地图上
这里使用的是苹果自带的地图
- (void)showLocationOnMapWithLatitude:(CLLocationDegrees)latitude longitude:(CLLocationDegrees)longitude {//创建坐标点CLLocationCoordinate2D coordinate = CLLocationCoordinate2DMake(latitude, longitude);//以指定的坐标为中心,设置地图显示范围。这里的参数 1000 表示地图的纬度和经度跨度都为 1000 米。MKCoordinateRegion region = MKCoordinateRegionMakeWithDistance(coordinate, 1000, 1000);//调整地图显示区域,region 是一个 MKCoordinateRegion 结构体,表示地图的中心点和跨度,这里 region 代表指定的坐标点为中心[self.mapView setRegion:region animated:YES];// 添加标注//MKPointAnnotation是 MapKit 框架中的一部分,用于表示地图上的点标注MKPointAnnotation *annotation = [[MKPointAnnotation alloc] init];//将coordinate属性设置为当前经纬度坐标annotation.coordinate = coordinate;//为地图添加标记[self.mapView addAnnotation:annotation];
}
运行结果: