講解iOS開發(fā)中基本的定位功能實現(xiàn)
一、簡單說明
1.CLLocationManager
CLLocationManager的常用操作和屬性
開始用戶定位- (void)startUpdatingLocation;
停止用戶定位- (void) stopUpdatingLocation;
說明:當(dāng)調(diào)用了startUpdatingLocation方法后,就開始不斷地定位用戶的位置,中途會頻繁地調(diào)用代理的下面方法
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations;
每隔多少米定位一次
@property(assign, nonatomic) CLLocationDistance distanceFilter;
定位精確度(越精確就越耗電)
@property(assign, nonatomic) CLLocationAccuracy desiredAccuracy;
2.CLLocation
CLLocation用來表示某個位置的地理信息,比如經(jīng)緯度、海拔等等
(1)經(jīng)緯度
@property(readonly, nonatomic) CLLocationCoordinate2D coordinate;
(2)海拔
@property(readonly, nonatomic) CLLocationDistance altitude;
(3)路線,航向(取值范圍是0.0° ~ 359.9°,0.0°代表真北方向)
@property(readonly, nonatomic) CLLocationDirection course;
(4)行走速度(單位是m/s)
@property(readonly, nonatomic) CLLocationSpeed speed;
(5)計算2個位置之間的距離
- (CLLocationDistance)distanceFromLocation:(const CLLocation *)location方法
3.CLLocationCoordinate2D
CLLocationCoordinate2D是一個用來表示經(jīng)緯度的結(jié)構(gòu)體,定義如下
typedef struct {
CLLocationDegrees latitude; // 緯度
CLLocationDegrees longitude; // 經(jīng)度
} CLLocationCoordinate2D;
一般用CLLocationCoordinate2DMake函數(shù)來創(chuàng)建CLLocationCoordinate2D
二、代碼示例
//
// YYViewController.m
// 18-定位服務(wù)
//
// Created by apple on 14-8-9.
// Copyright (c) 2014年 yangyong. All rights reserved.
//
#import "YYViewController.h"
#import <CoreLocation/CoreLocation.h>
//需要遵守CLLocationManagerDelegate協(xié)議
@interface YYViewController ()<CLLocationManagerDelegate>
@property(nonatomic,strong)CLLocationManager *locMgr;
@end
@implementation YYViewController
#pragma mark-懶加載
-(CLLocationManager *)locMgr
{
if (_locMgr==nil) {
//1.創(chuàng)建位置管理器(定位用戶的位置)
self.locMgr=[[CLLocationManager alloc]init];
//2.設(shè)置代理
self.locMgr.delegate=self;
}
return _locMgr;
}
- (void)viewDidLoad
{
[super viewDidLoad];
//判斷用戶定位服務(wù)是否開啟
if ([CLLocationManager locationServicesEnabled]) {
//開始定位用戶的位置
[self.locMgr startUpdatingLocation];
//每隔多少米定位一次(這里的設(shè)置為任何的移動)
self.locMgr.distanceFilter=kCLDistanceFilterNone;
//設(shè)置定位的精準(zhǔn)度,一般精準(zhǔn)度越高,越耗電(這里設(shè)置為精準(zhǔn)度最高的,適用于導(dǎo)航應(yīng)用)
self.locMgr.desiredAccuracy=kCLLocationAccuracyBestForNavigation;
}else
{//不能定位用戶的位置
//1.提醒用戶檢查當(dāng)前的網(wǎng)絡(luò)狀況
//2.提醒用戶打開定位開關(guān)
}
//測試方法,計算兩個位置之間的距離
[self countDistance];
}
#pragma mark-CLLocationManagerDelegate
/**
* 當(dāng)定位到用戶的位置時,就會調(diào)用(調(diào)用的頻率比較頻繁)
*/
-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations
{
//locations數(shù)組里邊存放的是CLLocation對象,一個CLLocation對象就代表著一個位置
CLLocation *loc = [locations firstObject];
//維度:loc.coordinate.latitude
//經(jīng)度:loc.coordinate.longitude
NSLog(@"緯度=%f,經(jīng)度=%f",loc.coordinate.latitude,loc.coordinate.longitude);
NSLog(@"%d",locations.count);
//停止更新位置(如果定位服務(wù)不需要實時更新的話,那么應(yīng)該停止位置的更新)
// [self.locMgr stopUpdatingLocation];
}
//計算兩個位置之間的距離
-(void)countDistance
{
//根據(jù)經(jīng)緯度創(chuàng)建兩個位置對象
CLLocation *loc1=[[CLLocation alloc]initWithLatitude:40 longitude:116];
CLLocation *loc2=[[CLLocation alloc]initWithLatitude:41 longitude:116];
//計算兩個位置之間的距離
CLLocationDistance distance=[loc1 distanceFromLocation:loc2];
NSLog(@"(%@)和(%@)的距離=%fM",loc1,loc2,distance);
}
@end
打印查看:

代碼說明:
1.關(guān)于代理方法
需要設(shè)置代理,通過代理告訴用戶當(dāng)前的位置,有兩個代理方法:
locations參數(shù)里面裝著CLLocation對象
其中后者是一個過期的方法,在新的方法(第一個)中使用了一個數(shù)組來替代。
說明:該方法在當(dāng)定位到用戶的位置時就會調(diào)用,調(diào)用比較頻繁
注意:不要使用局部變量(創(chuàng)建位置管理器),因為局部變量的方法結(jié)束它就被銷毀了。建議使用一個全局的變量,且只創(chuàng)建一次就可以了(使用懶加載)。
2.定位的精度
3.如果發(fā)現(xiàn)自己的定位服務(wù)沒有打開,那么應(yīng)該提醒用戶打開定位服務(wù)功能。
4.定位服務(wù)是比較耗電的,如果是做定位服務(wù)(沒必要實時更新的話),那么定位了用戶位置后,應(yīng)該停止更新位置。
三、用戶隱私的保護
1.權(quán)限設(shè)置說明
從iOS 6開始,蘋果在保護用戶隱私方面做了很大的加強,以下操作都必須經(jīng)過用戶批準(zhǔn)授權(quán)
(1)要想獲得用戶的位置
(2)想訪問用戶的通訊錄、日歷、相機、相冊等
當(dāng)想訪問用戶的隱私信息時,系統(tǒng)會自動彈出一個對話框讓用戶授權(quán)
注意:一旦用戶選擇了“Don't Allow”,意味著你的應(yīng)用以后就無法使用定位功能,且當(dāng)用戶第一次選擇了之后,以后就再也不會提醒進行設(shè)置。
因此在程序中應(yīng)該進行判斷,如果發(fā)現(xiàn)自己的定位服務(wù)沒有打開,那么應(yīng)該提醒用戶打開定位服務(wù)功能。
CLLocationManager有個類方法可以判斷當(dāng)前應(yīng)用的定位功能是否可用+ (BOOL)locationServicesEnabled;
常用的方法:截圖告訴用戶,應(yīng)該怎么打開授權(quán)
2.開發(fā)者可以在Info.plist中設(shè)置NSLocationUsageDescription說明定位的目的(Privacy - Location Usage Description)
說明:這里的定位服務(wù)是基于網(wǎng)絡(luò)的。通常定位服務(wù)可以是基于GPS、基站或者是網(wǎng)絡(luò)的。
四、iOS8以來的改進
iOS 8 還提供了更加人性化的定位服務(wù)選項。App 的定位服務(wù)不再僅僅是關(guān)閉或打開,現(xiàn)在,定位服務(wù)的啟用提供了三個選項,「永不」「使用應(yīng)用程序期間」和「始終」。同時,考慮到能耗問題,如果一款 App 要求始終能在后臺開啟定位服務(wù),iOS 8 不僅會在首次打開 App 時主動向你詢問,還會在日常使用中彈窗提醒你該 App 一直在后臺使用定位服務(wù),并詢問你是否繼續(xù)允許。在iOS7及以前的版本,如果在應(yīng)用程序中使用定位服務(wù)只要在程序中調(diào)用startUpdatingLocation方法應(yīng)用就會詢問用戶是否允許此應(yīng)用是否允許使用定位服務(wù),同時在提示過程中可以通過在info.plist中配置通過配置Privacy - Location Usage Description告訴用戶使用的目的,同時這個配置是可選的。
但是在iOS8中配置配置項發(fā)生了變化,可以通過配置NSLocationAlwaysUsageDescription或者NSLocationWhenInUseUsageDescription來告訴用戶使用定位服務(wù)的目的,并且注意這個配置是必須的,如果不進行配置則默認(rèn)情況下應(yīng)用無法使用定位服務(wù),打開應(yīng)用不會給出打開定位服務(wù)的提示,除非安裝后自己設(shè)置此應(yīng)用的定位服務(wù)。同時,在應(yīng)用程序中需要根據(jù)配置對requestAlwaysAuthorization或locationServicesEnabled方法進行請求。由于本人機器已經(jīng)更新到最新的iOS8.1下面的內(nèi)容主要針對iOS8,使用iOS7的朋友需要稍作調(diào)整。
//
// KCMainViewController.m
// CoreLocation
//
// Created by Kenshin Cui on 14-03-27.
// Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//
#import "KCMainViewController.h"
#import <CoreLocation/CoreLocation.h>
@interface KCMainViewController ()<CLLocationManagerDelegate>{
CLLocationManager *_locationManager;
}
@end
@implementation KCMainViewController
- (void)viewDidLoad {
[super viewDidLoad];
//定位管理器
_locationManager=[[CLLocationManager alloc]init];
if (![CLLocationManager locationServicesEnabled]) {
NSLog(@"定位服務(wù)當(dāng)前可能尚未打開,請設(shè)置打開!");
return;
}
//如果沒有授權(quán)則請求用戶授權(quán)
if ([CLLocationManager authorizationStatus]==kCLAuthorizationStatusNotDetermined){
[_locationManager requestWhenInUseAuthorization];
}else if([CLLocationManager authorizationStatus]==kCLAuthorizationStatusAuthorizedWhenInUse){
//設(shè)置代理
_locationManager.delegate=self;
//設(shè)置定位精度
_locationManager.desiredAccuracy=kCLLocationAccuracyBest;
//定位頻率,每隔多少米定位一次
CLLocationDistance distance=10.0;//十米定位一次
_locationManager.distanceFilter=distance;
//啟動跟蹤定位
[_locationManager startUpdatingLocation];
}
}
#pragma mark - CoreLocation 代理
#pragma mark 跟蹤定位代理方法,每次位置發(fā)生變化即會執(zhí)行(只要定位到相應(yīng)位置)
//可以通過模擬器設(shè)置一個虛擬位置,否則在模擬器中無法調(diào)用此方法
-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations{
CLLocation *location=[locations firstObject];//取出第一個位置
CLLocationCoordinate2D coordinate=location.coordinate;//位置坐標(biāo)
NSLog(@"經(jīng)度:%f,緯度:%f,海拔:%f,航向:%f,行走速度:%f",coordinate.longitude,coordinate.latitude,location.altitude,location.course,location.speed);
//如果不需要實時定位,使用完即使關(guān)閉定位服務(wù)
[_locationManager stopUpdatingLocation];
}
@end
注意:
1.定位頻率和定位精度并不應(yīng)當(dāng)越精確越好,需要視實際情況而定,因為越精確越耗性能,也就越費電。
2.定位成功后會根據(jù)設(shè)置情況頻繁調(diào)用-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations方法,這個方法返回一組地理位置對象數(shù)組,每個元素一個CLLocation代表地理位置信息(包含經(jīng)度、緯度、海報、行走速度等信息),之所以返回數(shù)組是因為有些時候一個位置點可能包含多個位置。
3.使用完定位服務(wù)后如果不需要實時監(jiān)控應(yīng)該立即關(guān)閉定位服務(wù)以節(jié)省資源。
4.除了提供定位功能,CLLocationManager還可以調(diào)用startMonitoringForRegion:方法對指定區(qū)域進行監(jiān)控。
相關(guān)文章
iOS 隱藏導(dǎo)航條和狀態(tài)欄實現(xiàn)方法
這篇文章主要介紹了 iOS隱藏導(dǎo)航條和狀態(tài)欄實現(xiàn)方法的相關(guān)資料,有時候根據(jù)需求開發(fā)APP 需要隱藏導(dǎo)航欄和狀態(tài)欄,這里提供了實現(xiàn)方法需要的朋友可以參考下2016-11-11ios使用NSProxy實現(xiàn)消息轉(zhuǎn)發(fā)
本文主要介紹了ios使用NSProxy實現(xiàn)消息轉(zhuǎn)發(fā),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-07-07iOS App中UITableView左滑出現(xiàn)刪除按鈕及其cell的重用
這篇文章主要介紹了iOS App中UITableView左滑出現(xiàn)刪除按鈕及其cell的重用的方法,實例代碼為傳統(tǒng)的Objective-C語言,需要的朋友可以參考下2016-03-03Flutter Widgets MediaQuery控件屏幕信息適配
這篇文章主要為大家介紹了Flutter Widgets 之 MediaQuery控件獲取屏幕信息和屏幕適配示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-11-11