知識簡介
-
在移動互聯(lián)網(wǎng)時代,移動app能解決用戶的很多生活瑣事,比如
- 導航:去任意陌生的地方
- 周邊:找餐館、找酒店、找銀行、找電影院
-
在上述應用中,都用到了地圖和定位功能,在iOS開發(fā)中,要想加入這2大功能,必須基于2個框架進行開發(fā)
- Map Kit :用于地圖展示
- Core Location :用于地理定位
-
2個熱門專業(yè)術語
- LBS :Location Based Service
- SoLoMo :Social Local Mobile(索羅門)
經(jīng)緯度基本知識

經(jīng)緯度基本知識.png
如何設置模擬器的經(jīng)緯度

設置模擬器經(jīng)緯度.png
- 例如設置為杭州經(jīng)緯度

杭州經(jīng)緯度.png
用戶隱私的保護
-
從iOS 6開始,蘋果在保護用戶隱私方面做了很大的加強,以下操作都必須經(jīng)過用戶批準授權
- 要想獲得用戶的位置
- 想訪問用戶的通訊錄、日歷、相機、相冊等等
當想訪問用戶的隱私信息時,系統(tǒng)會自動彈出一個對話框讓用戶授權

請求對話框.png
注意:一旦用戶選擇了“Don’t Allow”,意味著你的應用以后就無法使用定位功能
為了嚴謹起見,最好在使用定位功能之前判斷當前應用的定位功能是否可用
CLLocationManager有個類方法可以判斷當前應用的定位功能是否可用
+ (BOOL)locationServicesEnabled;
從iOS 8開始,用戶定位分兩種情況
總是使用用戶位置: NSLocationAlwaysUsageDescription
使用應用時定位: NSLocationWhenInUseDescription
需要在plist中添加字段

plist中添加字段.png
CoreLocation框架的使用
- 導入主頭文件
#import <CoreLocation/CoreLocation.h>
- CoreLocation框架使用須知
- CoreLocation框架中所有數(shù)據(jù)類型的前綴都是CL
- CoreLocation中使用CLLocationManager對象來做用戶定位
CLLocationManager的常用屬性及操作
- 每隔多少米定位一次
@property(assign, nonatomic) CLLocationDistance distanceFilter;
- 定位精確度(越精確就越耗電)
@property(assign, nonatomic) CLLocationAccuracy desiredAccuracy;
- 開始用戶定位
- (void)startUpdatingLocation;
- 停止用戶定位
- (void) stopUpdatingLocation;
- 當調(diào)用了startUpdatingLocation方法后,就開始不斷地定位用戶的位置,中途會頻繁地調(diào)用代理的下面方法
// locations參數(shù)里面裝著CLLocation對象
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations;
CLLocation用來表示某個位置的地理信息,比如經(jīng)緯度、海拔等等
- 經(jīng)緯度
@property(readonly, nonatomic) CLLocationCoordinate2D coordinate;
- 海拔
@property(readonly, nonatomic) CLLocationDistance altitude;
- 路線,航向(取值范圍是0.0° ~ 359.9°,0.0°代表真北方向)
@property(readonly, nonatomic) CLLocationDirection course;
- 行走速度(單位是m/s)
@property(readonly, nonatomic) CLLocationSpeed speed;
- 計算2個位置之間的距離
- (CLLocationDistance)distanceFromLocation:(const CLLocation *)location
CoreLocation基本使用
- 獲取當前位置經(jīng)緯度
- 計算2個位置之間的距離
實現(xiàn)步驟
1.獲取用戶的授權狀態(tài)-->請求授權(info.plist-->NSLocationAlwaysUsageDescription/string)
2.請求用戶的位置
- 創(chuàng)建CLLocationManager
- 設置代理
- 實現(xiàn)代理方法
- 請求獲取用戶的位置(startUpdatingLocation)
3.獲取用戶的位置(CLLocation)
- coordinate-->latitude/longitude
4.設置定位精確度desiredAccuracy/設置當用戶移動多少距離,重新定位distanceFilter
5.計算兩個經(jīng)緯度之間距離(包裝CLLocation對象-->distanceFromLocation)
#######代碼如下
#import "ViewController.h"
#import <CoreLocation/CoreLocation.h>
@interface ViewController () <CLLocationManagerDelegate>
/** 用戶位置管理者對象 */
@property (nonatomic, strong) CLLocationManager *mgr;
@end
@implementation ViewController
#pragma mark - 懶加載
- (CLLocationManager *)mgr
{
if (_mgr == nil) {
self.mgr = [[CLLocationManager alloc] init];
// 設置代理,在代理方法中可以拿到用戶的位置
self.mgr.delegate = self;
// 設置定位的精度(精度越高越耗電)
self.mgr.desiredAccuracy = kCLLocationAccuracyBestForNavigation;
// 設置當用戶移動的時候,重新來定位
self.mgr.distanceFilter = 10.0;
}
return _mgr;
}
- (void)viewDidLoad {
[super viewDidLoad];
// 1.獲取用戶的授權狀態(tài)(iOS7只要使用到定位,就會直接請求授權)
CLAuthorizationStatus status = [CLLocationManager authorizationStatus];
if (status == kCLAuthorizationStatusNotDetermined) {
/*
if ([[UIDevice currentDevice].systemVersion doubleValue] >= 8.0) {
[mgr requestAlwaysAuthorization];
}
*/
if ([self.mgr respondsToSelector:@selector(requestAlwaysAuthorization)]) {
[self.mgr requestAlwaysAuthorization];
}
}
// 2.開始定位(當調(diào)用該方法,系統(tǒng)就會不停的更新用戶的位置)
[self.mgr startUpdatingLocation];
// 3.計算兩個經(jīng)緯度之間的距離
[self countDistance];
}
- (void)countDistance
{
// 北京:39.6 116.39
// 杭州:30.3 120.2
CLLocation *location1 = [[CLLocation alloc] initWithLatitude:39.6 longitude:116.39];
CLLocation *location2 = [[CLLocation alloc] initWithLatitude:30.3 longitude:120.2];
// 計算距離
CLLocationDistance distance = [location1 distanceFromLocation:location2];
NSLog(@"%.2f", distance);
}
#pragma mark - 實現(xiàn)CLLocationManager的代理方法
/**
* 當獲取到用戶的位置就會執(zhí)行該方法(如果僅僅是想拿到用戶的位置,可以在獲取到用戶位置之后停止停止定位)
*
* @param locations 數(shù)組中就存放著用戶的位置(每更新到用戶的一個位置,就會將用戶位置的對象加入數(shù)組中)
*/
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations
{
/*
@property(readonly, nonatomic) CLLocationDistance altitude; // 海拔
@property(readonly, nonatomic) CLLocationAccuracy horizontalAccuracy; // 水平精度
@property(readonly, nonatomic) CLLocationAccuracy verticalAccuracy; // 垂直精度@property(readonly, nonatomic) CLLocationSpeed speed 手機當前的速度
@property(readonly, nonatomic, copy) NSDate *timestamp; 時間戳
@property(readonly, nonatomic, copy) CLFloor *floor 樓層
*/
// 1.拿到用戶位置的對象
CLLocation *location = [locations lastObject];
// 2.拿到用戶當前位置的經(jīng)緯度
CLLocationCoordinate2D coordinate = location.coordinate;
NSLog(@"latitude = %.2f", coordinate.latitude);
NSLog(@"longitude = %.2f", coordinate.longitude);
// [manager stopUpdatingLocation];
}
@end
注意點:使用模擬器獲取當前位置時可能出現(xiàn)的不打印的情況,可能是以下原因
- 未設置模擬器經(jīng)緯度(設置方法在上文中)
- 模擬器自身問題(更換模擬器比如6s換成5s,最好還是用真機測試)
地理編碼與反地理編碼
CLGeocoder
- 使用CLGeocoder可以完成“地理編碼”和“反地理編碼”
- 地理編碼:根據(jù)給定的地名,獲得具體的位置信息(比如經(jīng)緯度、地址的全稱等)
- 反地理編碼:根據(jù)給定的經(jīng)緯度,獲得具體的位置信息
地理編碼方法
- (void)geocodeAddressString:(NSString *)addressString completionHandler:(CLGeocodeCompletionHandler)completionHandler;
反地理編碼方法
- (void)reverseGeocodeLocation:(CLLocation *)location completionHandler:(CLGeocodeCompletionHandler)completionHandler;
- 當?shù)乩韁反地理編碼完成時,就會調(diào)用
CLGeocodeCompletionHandler
typedef void (^CLGeocodeCompletionHandler)(NSArray *placemarks, NSError *error);
- 這個block傳遞2個參數(shù)
- error :當編碼出錯時(比如編碼不出具體的信息)有值
- placemarks :里面裝著CLPlacemark對象
CLPlacemark
- CLPlacemark的字面意思是地標,封裝詳細的地址位置信息
地理位置
@property (nonatomic, readonly) CLLocation *location;
- 區(qū)域
@property (nonatomic, readonly) CLRegion *region;
- 詳細的地址信息
@property (nonatomic, readonly) NSDictionary *addressDictionary;
- 地址名稱
@property (nonatomic, readonly) NSString *name;
- 城市
@property (nonatomic, readonly) NSString *locality;
示例

布局.png
#import "ViewController.h"
#import <CoreLocation/CoreLocation.h>
@interface ViewController ()
// 地理編碼
@property (weak, nonatomic) IBOutlet UITextField *textField;
// 反地理編碼
@property (weak, nonatomic) IBOutlet UITextField *longitudeText;
@property (weak, nonatomic) IBOutlet UITextField *latitudeText;
/**編碼使用的對象*/
@property (strong, nonatomic) CLGeocoder *geoCoder;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
}
// 反地理編碼
- (IBAction)geoCoder:(id)sender {
if (self.textField.text.length == 0) {
NSLog(@"請輸入詳細信息");
return;
}
// 地理編碼
[self.geoCoder geocodeAddressString:self.textField.text completionHandler:^(NSArray<CLPlacemark *> * _Nullable placemarks, NSError * _Nullable error) {
if (error || placemarks.count == 0) {
return ;
}
for (CLPlacemark *pm in placemarks) {
CLLocationCoordinate2D coordinate = pm.location.coordinate;
// 獲取經(jīng)緯度
NSLog(@"%.2f", coordinate.latitude);
NSLog(@"%.2f", coordinate.longitude);
// 獲取所在的省
NSLog(@"%@", pm.administrativeArea);
// 獲取詳細信息(省、市)
NSLog(@"%@", pm.name);
}
}];
}
// 反地理編碼
- (IBAction)reverseGeocoder:(id)sender {
NSString *latitude = self.latitudeText.text;
NSString *longitude = self.longitudeText.text;
if (latitude.length == 0 || longitude.length == 0 ){
NSLog(@"請輸入經(jīng)緯度");
return;
}
CLLocation *location = [[CLLocation alloc] initWithLatitude:latitude.doubleValue longitude:longitude.doubleValue];
[self.geoCoder reverseGeocodeLocation:location completionHandler:^(NSArray<CLPlacemark *> * _Nullable placemarks, NSError * _Nullable error) {
for (CLPlacemark *pm in placemarks) {
// 獲取地址的全稱
NSLog(@"%@", pm.name);
// 獲取經(jīng)緯度
CLLocationCoordinate2D coordinate = pm.location.coordinate;
NSLog(@"緯度:%.2f", coordinate.latitude);
NSLog(@"經(jīng)度:%.2f", coordinate.longitude);
// 獲取城市
NSLog(@"所在城市:%@", pm.administrativeArea);
NSLog(@"所在城市:%@", pm.locality);
}
}];
}
#pragma mark - 懶加載
- (CLGeocoder *)geoCoder
{
if (_geoCoder == nil) {
_geoCoder = [[CLGeocoder alloc] init];
}
return _geoCoder;
}
@end