現(xiàn)在的移動(dòng)設(shè)備很多都提供定位服務(wù),使用iOS系統(tǒng)的iPhone、iPod Touch和iPad都可以提供位置服務(wù),iOS設(shè)備能提供3種不同途徑進(jìn)行定位:Wifi, 蜂窩式移動(dòng)電話(huà)基站, GPS衛(wèi)星
創(chuàng)新互聯(lián)建站自成立以來(lái),一直致力于為企業(yè)提供從網(wǎng)站策劃、網(wǎng)站設(shè)計(jì)、網(wǎng)站設(shè)計(jì)制作、網(wǎng)站制作、電子商務(wù)、網(wǎng)站推廣、網(wǎng)站優(yōu)化到為企業(yè)提供個(gè)性化軟件開(kāi)發(fā)等基于互聯(lián)網(wǎng)的全面整合營(yíng)銷(xiāo)服務(wù)。公司擁有豐富的網(wǎng)站建設(shè)和互聯(lián)網(wǎng)應(yīng)用系統(tǒng)開(kāi)發(fā)管理經(jīng)驗(yàn)、成熟的應(yīng)用系統(tǒng)解決方案、優(yōu)秀的網(wǎng)站開(kāi)發(fā)工程師團(tuán)隊(duì)及專(zhuān)業(yè)的網(wǎng)站設(shè)計(jì)師團(tuán)隊(duì)。
iOS不像Android系統(tǒng)在定位服務(wù)編程時(shí),可以指定采用哪種途徑進(jìn)行定位。iOS的API把底層這些細(xì)節(jié)屏蔽掉了,開(kāi)發(fā)人員和用戶(hù)并不知道現(xiàn)在設(shè)備是采用哪種方式進(jìn)行定位的,iOS系統(tǒng)會(huì)根據(jù)設(shè)備的情況和周?chē)沫h(huán)境,采用一套最佳的解決方案。這個(gè)方案是這樣的,如果能夠接收GPS信息,那么設(shè)備優(yōu)先采用GPS定位,否則采用Wifi或蜂窩基站定位,在Wifi和蜂窩基站之間優(yōu)先使用Wifi,如果無(wú)法連接Wifi才使用蜂窩基站定位。
總體來(lái)說(shuō)GPS定位優(yōu)點(diǎn)是準(zhǔn)確、覆蓋面廣闊,缺點(diǎn)是不能被遮擋(例如:在建筑物里面收不到GPS衛(wèi)星信號(hào))、GPS開(kāi)啟后比較費(fèi)電。蜂窩基站不僅誤差比較大,而且會(huì)耗費(fèi)用戶(hù)流量費(fèi)。而Wifi定位是最經(jīng)濟(jì)實(shí)惠的。
定位服務(wù)編程
定位服務(wù)在iOS 6之后API沒(méi)有太大的變化,主要使用CoreLocation框架,定位時(shí)候主要使用CLLocationManager、CLLocationManagerDelegate和CLLocation。CLLocationManager是定位服務(wù)管理類(lèi)它能夠給我們提供獲得設(shè)備的位置信息和高度信息,也可以監(jiān)控設(shè)備進(jìn)入或離開(kāi)某個(gè)區(qū)域,它還可以幫助獲得設(shè)備的運(yùn)行方向等。CLLocationManagerDelegate是CLLocationManager類(lèi)委托協(xié)議。CLLocation類(lèi)是封裝了位置和高度信息。
在定位服務(wù)的應(yīng)用中,第一次請(qǐng)求獲得位置信息時(shí)候,系統(tǒng)會(huì)提示用戶(hù)是否允許開(kāi)啟定位服務(wù)。用戶(hù)所在的位置是比較私密的信息,應(yīng)用獲取這些信息用戶(hù)是有知情權(quán)和否定權(quán)的。如果應(yīng)用在用戶(hù)不知情的情況下,而獲得用戶(hù)的位置信息,這在某些國(guó)家是違法的行為。
選擇“不允許”,定位服務(wù)就無(wú)法獲得位置信息了,如果想改變這些設(shè)置可以在系統(tǒng)設(shè)置應(yīng)用中開(kāi)啟或關(guān)閉。
我們可以關(guān)閉所有的定位服務(wù),只需要把最上面的“定位服務(wù)”開(kāi)關(guān)控件關(guān)閉就可以了。下面的具體應(yīng)用也可以關(guān)閉和開(kāi)啟。
下面我們通過(guò)一個(gè)案例介紹一下使用定位服務(wù)編程,在應(yīng)用啟動(dòng)時(shí)候啟動(dòng),進(jìn)入畫(huà)面時(shí)候會(huì)獲得位置信息,并顯示在對(duì)應(yīng)的文本框中,如果設(shè)備位置發(fā)送變化,也會(huì)重新會(huì)的位置信息,并更新對(duì)應(yīng)的文本框。
首先要實(shí)現(xiàn)定位服務(wù)的案例,需要為工程引入CoreLocation框架,添加具體步驟是選擇工程中的TARGETS→WhereAmI→Build Phases→Link Binary With Libraries,選擇右下角的“+”按鈕,打開(kāi)框架和庫(kù)選擇對(duì)話(huà)框
再添加對(duì)話(huà)框中選擇CoreLocation.framework,點(diǎn)擊Add按鈕后添加完成。UI設(shè)計(jì)部分我們不再介紹。我們直接看看實(shí)現(xiàn)代碼,其中主要代碼是視圖控制器ViewController中編寫(xiě)的,其中ViewController.h代碼如下:
#import#import #import @interface ViewController : UIViewController //經(jīng)度 @property (weak, nonatomic) IBOutlet UITextField *txtLng; //緯度 @property (weak, nonatomic) IBOutlet UITextField *txtLat; //高度 @property (weak, nonatomic) IBOutlet UITextField *txtAlt; @property(nonatomic, strong) CLLocationManager *locationManager; @end
在h文件中首先需要引入
ViewController.m的viewDidLoad代碼如下:
- (void)viewDidLoad { [super viewDidLoad]; //定位服務(wù)管理對(duì)象初始化 _locationManager = [[CLLocationManager alloc] init]; _locationManager.delegate = self; _locationManager.desiredAccuracy = kCLLocationAccuracyBest; ① _locationManager.distanceFilter = 1000.0f; ② }
在viewDidLoad方法中,主要對(duì)CLLocationManager的成員變量_locationManager進(jìn)行初始化。首先使用[[CLLocationManager alloc] init]語(yǔ)句實(shí)例化CLLocationManager對(duì)象。然后_locationManager.delegate = self語(yǔ)句設(shè)置定位服務(wù)委托為self。第①行代碼設(shè)置desiredAccuracy屬性,它是一個(gè)非常重要的屬性,它的取值有6個(gè)常量:kCLLocationAccuracyNearestTenMeters。精度10米;kCLLocationAccuracyHundredMeters 。精度100米;kCLLocationAccuracyKilometer 。精度1000米;kCLLocationAccuracyThreeKilometers。精度3000米;kCLLocationAccuracyBest 。設(shè)備使用電池供電時(shí)候,最高的精度;kCLLocationAccuracyBestForNavigation。導(dǎo)航情況下最高精度,一般要有外接電源時(shí)才能使用;
精度越高請(qǐng)求獲得位置信息的時(shí)間就越短,這就意味著設(shè)備越耗電。因此一個(gè)應(yīng)用應(yīng)該選擇適合它的精度,如果你的應(yīng)用是一個(gè)車(chē)載導(dǎo)航應(yīng)用,kCLLocationAccuracyBestForNavigation是比較好的選擇,你可以使用汽車(chē)上的電瓶為設(shè)備供電。如果你的應(yīng)用為徒步旅行者提供的導(dǎo)航應(yīng)用,kCLLocationAccuracyHundredMeters是一個(gè)不錯(cuò)的選擇。
第②行代碼設(shè)置distanceFilter屬性,它是距離過(guò)濾器,它定義了設(shè)備移動(dòng)更新位置信息的最小距離,它的單位是米,本例設(shè)置了1000米。
初始化CLLocationManager完成之后,需要使用startUpdatingLocation方法開(kāi)始定位服務(wù)。它是在ViewController.m的viewWillAppear:方法中,代碼如下:
- (void)viewWillAppear:(BOOL)animated { [super viewWillAppear:animated]; //開(kāi)始定位 [_locationManager startUpdatingLocation]; }
調(diào)用startUpdatingLocation方法定位服務(wù)就會(huì)開(kāi)啟,它根據(jù)設(shè)定的條件,不斷請(qǐng)求回調(diào)新的位置信息。因此開(kāi)啟這個(gè)方法一定要慎重,要在最合適的時(shí)候開(kāi)啟,在視圖控制器的聲明周期方法中viewWillAppear:是最合適的。與開(kāi)啟服務(wù)對(duì)應(yīng)的方法是stopUpdatingLocation方法,它的調(diào)用是在視圖控制器的viewWillDisappear:方法中調(diào)用的,代碼如下:
- (void)viewWillDisappear:(BOOL)animated { [super viewWillDisappear:animated]; //停止定位 [_locationManager stopUpdatingLocation]; }
viewWillDisappear:在視圖消失(應(yīng)用退到后臺(tái))時(shí)調(diào)用,能夠保證最及時(shí)地關(guān)閉定位服務(wù),這是負(fù)責(zé)任的做法。在iOS 6之后請(qǐng)求有所變化,定位服務(wù)應(yīng)用退入臺(tái)后可以延遲更新位置信息,其中allowDeferredLocationUpdatesUntilTraveled:timeout:方法可以設(shè)置延遲更新,從而使得應(yīng)用在后臺(tái)不再更新位置信息。關(guān)閉延遲更新使用disallowDeferredLocationUpdates方法實(shí)現(xiàn)。此外,在iOS 6之后新增pausesLocationUpdatesAutomatically屬性,它能設(shè)定自動(dòng)暫停位置更新,定位服務(wù)的開(kāi)啟和暫停管理權(quán)交給系統(tǒng),這樣會(huì)更加合理和簡(jiǎn)單。
一旦定位服務(wù)開(kāi)啟,并設(shè)置好了CLLocationManager委托屬性delegate后,當(dāng)用戶(hù)設(shè)備移動(dòng)到達(dá)過(guò)濾距離時(shí),就會(huì)回調(diào)委托方法,與定位服務(wù)有關(guān)的方法有兩個(gè):
locationManager:didUpdateLocations: 定位成功,是iOS 6新方法,替代之前的locationManager:didUpdateToLocation:fromLocation:方法;
locationManager:didFailWithError: 定位失敗;
實(shí)現(xiàn)CLLocationManager委托代碼如下:
#pragma mark Core Location委托方法用于實(shí)現(xiàn)位置的更新
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations { CLLocation * currLocation = [locations lastObject]; ① _txtLat.text = [NSString stringWithFormat:@"%3.5f", currLocation.coordinate.latitude]; ② _txtLng.text = [NSString stringWithFormat:@"%3.5f", currLocation.coordinate.longitude]; ③ _txtAlt.text = [NSString stringWithFormat:@"%3.5f", currLocation.altitude]; ④ } - (void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error { NSLog(@”error: %@”,error); }
在locationManager:didUpdateLocations:方法中參數(shù)locations是位置變化的集合,它按照時(shí)間變化的順序存放。如果想獲得當(dāng)前設(shè)備的位置,可以使用第①行的[locations lastObject]語(yǔ)句獲得集合中最后一個(gè)元素,它就是設(shè)備當(dāng)前位置了。從集合中返回的對(duì)象類(lèi)型是CLLocation,CLLocation封裝了位置、高度等信息。在上面代碼中我們使用了它的兩個(gè)屬性:altitude和coordinate,altitude屬性是高度值,coordinate是封裝了經(jīng)度和緯度的結(jié)構(gòu)體CLLocationCoordinate2D,CLLocationCoordinate2D定義如下:
typedef struct { CLLocationDegrees latitude; //緯度 CLLocationDegrees longitude; //經(jīng)度 } CLLocationCoordinate2D;
其中l(wèi)atitude為經(jīng)度信息,longitude為緯度信息,它們都是CLLocationDegrees類(lèi)型,CLLocationDegrees是使用typedef定義的double類(lèi)型。
第②行代碼中的newLocation.coordinate.latitude表達(dá)式是獲得設(shè)備當(dāng)前的緯度,第③行代碼中的newLocation.coordinate.longitude表達(dá)式是獲得設(shè)備當(dāng)前的緯度,而獲得高度可以使用第④行newLocation.altitude表達(dá)式直接獲得。