GPS(Gobal Positional System)全球定位系統,是一個中距離圓型軌道衛星導航系統,他可以為地球表面的絕大部分地區(98%)提供准備的定位、測速和高精度的時間標准。
Android支持地理定位服務的API。該地理定位服務可以用來獲取當前設備的地理位置,應用程序可以定時請求更新設備當前的地理定位信息。比如應用程序可以借助一個Intent接受器來實現如下功能:以經緯度和半徑劃定一個區域,當設備出入該區域時,發出提示信息,還可以和Google Map API一起使用,完成更多的任務。關於地理定位系統的API全部位於android.location包內,其中包括以下幾個重要的功能類:
類名
描述
LocationManager
提供訪問定位服務的功能,也提供獲取最佳定位提供者的功能,另外,臨時報警功能也可以借助該類來實現。
LocationProvider
定位提供者的抽象類。定位提供者具備周期性報告設備地理位置的功能。
LocationListener
提供定位信息發生改變時的回調共嫩。必須事先在定位管理器中注冊監聽器對象。
Criteria
使得應用能夠通過LocationProvider中設置的屬性來選擇合適的定位提供者。
Geocoder
用於處理地理編碼和反向地理編碼的類。地理編碼是指將地址或其他描述轉變為經度和緯度,反向地理編碼則是將經度和緯度轉變為地址或描述語言,其中包含了兩個構造函數,需要傳入經度和緯度的坐標。getFromLocation方法可以得到一組關於地址的數組。
要使用地理定位,首先需要取得LocationManager的實例,在Android中,獲得LocationManager的唯一方法是通過getSystemService方法的調用。通過使用LocationManager,我們可以獲得一個位置提供者的列表。在一個真實的手持設備中,這個列表包含了一些GPS服務。我們也可以選擇更強大、更精確、不帶其他附加服務的GPS。代碼如下:
LocationManager locationManager;
Stringcontext = Context.LOCATION_SERVICE;
locationManager= (LocationManager)getSystemService(context);
取得LocationManager對象之後,我們還需要注冊一個周期的更新視圖,代碼如下
LocationManager.requestLocationUpdate(LocationManager.GPS_PROVDER,1000, 0, locationListener);
其中第一個參數是設置服務提供者,第二個參數是周期,最後一個參數locationListener,是用來監聽定位信息的改變,必須要實現如下方法:
方法
描述
onLocationChanged(Location location)
當坐標改變時候觸發該函數,如果Provider傳相同的坐標,它就不會觸發。
onProviderDisabled(String provider)
Provider禁用時觸發此函數,比如GPS被關閉。
onProviderEnabled(String provider)
Provider啟用時觸發此函數,比如GPS被打開。
onStatusChanged(String provider, int status, Bundle extras)
Provider的轉態在可用、暫時不可用和無服務三個狀態直接切換時觸發此函數。
要使用定位的API,首先需要再AndroidManifest.xml文件中添加其權限,具體代碼如下:
- <uses-permission android:name="android.permission.INTERNET"/>
- <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
- <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
- <application
- android:icon="@drawable/ic_launcher"
- android:label="@string/app_name" >
-
- <uses-library android:name="com.google.android.maps"/>
-
- <activity
- android:name=".GPSActivity"
- android:label="@string/app_name" >
由於我們在模擬器上測試,所以需要人為的設置一個坐標。可用通過兩種方法來設置一個模擬的坐標值。第一種方法是通過DDMS,我們可用在Eclipse的ADT插件中使用這種方法,只要啟動Eclipse,選擇“Window”->“Show View”,打開“Emulator Control”界面手動或者通過KML和GPX文件來設置一個坐標。
另外一種方法使用geo命令,我們需要telnet到本機的5554端口,然後再命令行下輸入類似於geo fix-121.45365 46.51119 4392這樣的命令,後面三個參數分別是經度、緯度和(可選)海拔。設置後再Android模擬器屏幕上便多出了一個如圖9-17所示的標准,表示模擬了一個GPS權限。
現在我們可以使用位置管理器(LocationManager)和位置提供者進行getFromLocation的調用。這個方法放回本機當前位置的一個快照,這個快照將以Location對象形式提供。在手持設備中,我們可以獲得當前位置的經度和緯度;調用getFromLocationName方法可以返回一個數據表示一個地方的地名。
在這個地圖中,我們還可以創建了一個菜單來縮放地圖,這個功能是使用地圖控制器(MapController)的zoomIn和zoomOut方法來放大和縮小地圖。
下面試測試一個示例代碼:
- package cn.edu.pku;
-
- import java.io.IOException;
- import java.util.List;
- import java.util.Locale;
-
- import com.google.android.maps.GeoPoint;
- import com.google.android.maps.MapActivity;
- import com.google.android.maps.MapController;
- import com.google.android.maps.MapView;
- import com.google.android.maps.Overlay;
-
- import android.content.Context;
- import android.graphics.Bitmap;
- import android.graphics.BitmapFactory;
- import android.graphics.Canvas;
- import android.graphics.Paint;
- import android.graphics.Point;
- import android.location.Address;
- import android.location.Criteria;
- import android.location.Geocoder;
- import android.location.Location;
- import android.location.LocationListener;
- import android.location.LocationManager;
- import android.os.Bundle;
- import android.view.Menu;
- import android.view.MenuItem;
- import android.widget.TextView;
-
- public class GPSActivity extends MapActivity {
-
- public MapController mapController;
- public MyLocationOverlay myPosition;
- public MapView myMapView;
- public static final int ZOOM_IN = Menu.FIRST;
- public static final int ZOOM_OUT = Menu.FIRST + 1;
-
- @Override
- protected void onCreate(Bundle icicle) {
- // TODO Auto-generated method stub
- super.onCreate(icicle);
-
- setContentView(R.layout.main);
-
- LocationManager locationManager;
- String context = Context.LOCATION_SERVICE;
- locationManager = (LocationManager)getSystemService(context);
- myMapView = (MapView)findViewById(R.id.mapView1);
-
- mapController = myMapView.getController();
-
- //設置顯示模式
- myMapView.setSatellite(true);
- myMapView.setStreetView(true);
-
- //設置縮放控制,這裡使用自己實現的縮放菜單
- myMapView.displayZoomControls(false);
- //設置使用MyLocationOverlay繪圖
- mapController.setZoom(17);
- myPosition = new MyLocationOverlay();
- List<Overlay> overlays = myMapView.getOverlays();
- overlays.add(myPosition);
- //設置Criteria(服務商)的信息
- Criteria criteria = new Criteria();
- //經度要求
- criteria.setAccuracy(Criteria.ACCURACY_FINE);
- criteria.setAltitudeRequired(false);
- criteria.setBearingRequired(false);
- criteria.setCostAllowed(false);
- criteria.setPowerRequirement(Criteria.POWER_LOW);
- //取得最好效果的criteria
- String provider = locationManager.getBestProvider(criteria, true);
- //獲得坐標相應信息
- Location location = locationManager.getLastKnownLocation(provider);
- //更新坐標相關信息
- updateWithNewLocation(location);
- //注冊一個周期的更新,3000ms更新一次
- //locationManager用來監聽定位信息的改變
- locationManager.requestLocationUpdates(provider, 3000, 0, locationListener);
-
- }
-
- private void updateWithNewLocation(Location location){
- String latLongString;
- TextView myLocationText = (TextView)findViewById(R.id.textView1);
- String addressString = "沒有找到地址\n";
-
- if(location != null){
- //為繪制標志的類設置坐標
- //myPosition.
- //取得經度和緯度
- Double geoLat = location.getLatitude() * 1E6;
- Double geoLng = location.getLongitude() * 1E6;
-
- GeoPoint point = new GeoPoint(geoLat.intValue(), geoLng.intValue());
- //定位到指定坐標
- mapController.animateTo(point);
- double lat = location.getLatitude();
- double lng = location.getLongitude();
- latLongString = "經度:" + lat + "\n緯度:" + lng;
-
- double latitude = location.getLatitude();
- double longitude = location.getLongitude();
- //根據地理環境來確定編碼
- Geocoder gc = new Geocoder(this, Locale.getDefault());
- try{
- //取得地址相關的一些信息、經度、緯度
- List<Address> addresses = gc.getFromLocation(latitude, longitude, 1);
- StringBuilder sb = new StringBuilder();
- if(addresses.size() > 0){
- Address address = addresses.get(0);
- for(int i = 0; i < address.getMaxAddressLineIndex(); i++){
- sb.append(address.getAddressLine(i)).append("\n");
- sb.append(address.getLocality()).append("\n");
- sb.append(address.getPostalCode()).append("\n");
- sb.append(address.getCountryName());
- addressString = sb.toString();
- }
- }
- }catch(IOException e){}
- }else{
- latLongString = "沒有找到坐標. \n";
- }
-
- myLocationText.setText("你當前的坐標如下:\n" + latLongString + "\n" + addressString);
- }
-
- private final LocationListener locationListener = new LocationListener() {
-
- public void onStatusChanged(String provider, int status, Bundle extras) {//Provider轉態在可用、暫時不可服務和無服務三個狀態直接切換時觸發此函數
- // TODO Auto-generated method stub
-
- }
-
- public void onProviderEnabled(String provider) {//Provider啟用時觸發此函數,比如GPS被打開
- // TODO Auto-generated method stub
-
- }
-
- public void onProviderDisabled(String provider) {//Provider禁用時觸發此函數,比如GPS被關閉
- // TODO Auto-generated method stub
- updateWithNewLocation(null);
- }
-
- public void onLocationChanged(Location location) {//當坐標改變時觸發事件
- // TODO Auto-generated method stub
- updateWithNewLocation(location);
- }
- };
-
- @Override
- protected boolean isRouteDisplayed() {
- // TODO Auto-generated method stub
- return false;
- }
-
- @Override
- public boolean onOptionsItemSelected(MenuItem item) {
- // TODO Auto-generated method stub
- super.onOptionsItemSelected(item);
-
- switch(item.getItemId()){
- case ZOOM_IN:
- mapController.zoomIn();
- return true;
- case ZOOM_OUT:
- mapController.zoomOut();
- return true;
- }
-
- return true;
- }
-
- @Override
- public boolean onCreateOptionsMenu(Menu menu) {
- // TODO Auto-generated method stub
- super.onCreateOptionsMenu(menu);
-
- menu.add(0, ZOOM_IN, Menu.NONE, "放大");
- menu.add(0, ZOOM_OUT, Menu.NONE, "縮小");
- return true;
- }
-
- class MyLocationOverlay extends Overlay{
- Location mLocation;
- //更新坐標時,設置該坐標,以便畫圖
- public void setLocation(Location location){
- mLocation = location;
- }
-
- @Override
- public boolean draw(Canvas canvas, MapView mapView, boolean shadow,
- long when) {
- // TODO Auto-generated method stub
- super.draw(canvas, mapView, shadow, when);
-
- Paint paint = new Paint();
- Point myScreenCoords = new Point();
- //將經緯度轉換成實際屏幕坐標
- GeoPoint tmpGeoPoint = new GeoPoint((int)(mLocation.getLatitude() * 1E6), (int)(mLocation.getLongitude() * 1E6));
- mapView.getProjection().toPixels(tmpGeoPoint, myScreenCoords);
- paint.setStrokeWidth(1);
- paint.setARGB(255, 255, 0, 0);
- paint.setStyle(Paint.Style.STROKE);
- Bitmap bmp = BitmapFactory.decodeResource(getResources(), R.drawable.home);
- canvas.drawBitmap(bmp, myScreenCoords.x, myScreenCoords.y, paint);
- canvas.drawText("Here am I", myScreenCoords.x, myScreenCoords.y, paint);
-
- return true;
- }
- }
- }
運行效果:
注意:Loction常常獲取null,在網上查了很多資料。發現最主要是我們不能查到那個GPS提供商的能提供定位,有用while循環知道獲取停止,但是這個時間可能等待很長的時間都不能獲取到,我是采用下面的
- String provider = locationManager.getBestProvider(criteria, true);
- List<String> privatelist= locationManager.getAllProviders();
- for(String privates:privatelist)
- {
- Location locat=locationManager.getLastKnownLocation(privates);
- if(locat!=null)
- {
- provider=privates;
- break;
- }
- }
- //獲得坐標相應信息
- Location location = locationManager.getLastKnownLocation(provider);
這樣可以檢測到,但是這個不是最優的方法,但是可以得到運行的效果。