Android定位功能

不說廢話,直接說說實現android定位有關的API吧。

這些API都在android.location包下,一共有三個接口和八個類。它們配合使用即可實現定位功能。

三個接口:

GpsStatus.Listener:這是一個當GPS狀態發生改變時,用來接收通知的接口。

GpsStatus.NmeaListener:這是一個用來從GPS裡接收Nmea-0183(為海用電子設備制定的標準格式)信息的接口。

LocationListener:位置監聽器,用於接收當位置信息發生改變時從LocationManager接收通知的接口。

八個類:

Address:描述地址的類,比如:北京天安門

Criteria:用於描述Location Provider標準的類,標準包括位置精度水平,電量消耗水平,是否獲取海拔、方位信息,是否允許接收付費服務。

GeoCoder:用於處理地理位置的編碼。

GpsSatellite:和GpsStatus聯合使用,用於描述當前GPS衛星的狀態。

GpsStatus:和GpsStatus.Listener聯合使用,用於描述當前GPS衛星的狀態。

Location:用於描述位置信息。

LocationManager:通過此類獲取和調用系統位置服務

LocationProvider:用於描述Location Provider的抽象超類,一個LocationProvider應該能夠周期性的報告當前設備的位置信息。

這裡通過一個示例代碼來演示一下android定位。

首先,在AndroidManifest.xml清單文件裡需要加入ACCESS_FINE_LOCATION權限:

 

 package com.test;
 
import java.io.IOException;
import java.util.List;
 
import android.app.Activity;
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.util.Log;
import android.widget.Toast;
 
public class PositionActivity extends Activity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        // 獲取到LocationManager對象
        LocationManager locationManager = (LocationManager) getSystemService(LOCATION_SERVICE);
        // 創建一個Criteria對象
        Criteria criteria = new Criteria();
        // 設置粗略精確度
        criteria.setAccuracy(Criteria.ACCURACY_COARSE);
        // 設置是否需要返回海拔信息
        criteria.setAltitudeRequired(false);
        // 設置是否需要返回方位信息
        criteria.setBearingRequired(false);
        // 設置是否允許付費服務
        criteria.setCostAllowed(true);
        // 設置電量消耗等級
        criteria.setPowerRequirement(Criteria.POWER_HIGH);
        // 設置是否需要返回速度信息
        criteria.setSpeedRequired(false);
        // 根據設置的Criteria對象,獲取最符合此標準的provider對象 41
        String currentProvider = locationManager
                .getBestProvider(criteria, true);
        Log.d("Location", "currentProvider: " + currentProvider);
        // 根據當前provider對象獲取最後一次位置信息 44
        Location currentLocation = locationManager
                .getLastKnownLocation(currentProvider);
        // 如果位置信息為null,則請求更新位置信息 46
        if (currentLocation == null) {
            locationManager.requestLocationUpdates(currentProvider, 0, 0,
                    locationListener);
        }
        // 直到獲得最後一次位置信息為止,如果未獲得最後一次位置信息,則顯示默認經緯度 50
        // 每隔10秒獲取一次位置信息 51
        while (true) {
            currentLocation = locationManager
                    .getLastKnownLocation(currentProvider);
            if (currentLocation != null) {
                Log.d("Location", "Latitude: " + currentLocation.getLatitude());
                Log.d("Location", "location: " + currentLocation.getLongitude());
                break;
            } else {
                Log.d("Location", "Latitude: " + 0);
                Log.d("Location", "location: " + 0);
            }
            try {
                Thread.sleep(10000);
            } catch (InterruptedException e) {
                Log.e("Location", e.getMessage());
            }
        }
        // 解析地址並顯示 69
        Geocoder geoCoder = new Geocoder(this);
        try {
            int latitude = (int) currentLocation.getLatitude();
            int longitude = (int) currentLocation.getLongitude();
            List<Address> list = geoCoder.getFromLocation(latitude, longitude,
                    2);
            for (int i = 0; i < list.size(); i++) {
                Address address = list.get(i);
                Toast.makeText(
                        PositionActivity.this,
                        address.getCountryName() + address.getAdminArea()
                                + address.getFeatureName(), Toast.LENGTH_LONG)
                        .show();
            }
        } catch (IOException e) {
            Toast.makeText(PositionActivity.this, e.getMessage(),
                    Toast.LENGTH_LONG).show();
        }
    }
    // 創建位置監聽器 85
    private LocationListener locationListener = new LocationListener() {
        // 位置發生改變時調用 87
        @Override
        public void onLocationChanged(Location location) {
            Log.d("Location", "onLocationChanged");
            Log.d("Location",
                    "onLocationChanged Latitude" + location.getLatitude());
            Log.d("Location",
                    "onLocationChanged location" + location.getLongitude());
        }
 
        // provider失效時調用 95
        @Override
        public void onProviderDisabled(String provider) {
            Log.d("Location", "onProviderDisabled");
        }
 
        // provider啟用時調用101
        @Override
        public void onProviderEnabled(String provider) {
            Log.d("Location", "onProviderEnabled");
        }
 
        // 狀態改變時調用107
        @Override
        public void onStatusChanged(String provider, int status, Bundle extras) {
            Log.d("Location", "onStatusChanged");
        }
    };
}

由於代碼裡的Criteria對象對位置精度要求並不高,所以一般會返回“network”作為provider,而基於network的定位往往會存在一定的位置偏差,這對於需要精確定位的應用程序來說,顯然不合要求。這時,需要則需要用到基於GPS的定位方法瞭。具體請接著看文Android定位功能(二)

Android定位功能(二) (轉到第一部分)

在前文Android定位功能(一)中,已經大致介紹瞭一下在Android平臺中,和定位功能相關的類,並舉例獲取瞭位置信息。但是前文是基於Criteria定制瞭一個標準,通過getBestProvider()方法由Android系統自動獲取最符合Criteria的LocationProvider,從而實現瞭定位功能。這樣的做法能最大限度的保證定位功能的可實現性,但是卻無法保證獲取到的位置信息有最大的準確度。因為除瞭GPS外,其他定位方式都或多或少存在著位置偏移。

在實現GPS定位前,先瞭解一下GPS的部分特性:

1:GPS定位需要依靠3顆或以上的衛星。

2:GPS定位受環境影響較大,在晴朗的空地上,較容易搜索到衛星,而在室內通常是無法搜索到衛星的。

3:GPS定位需要使用GPS功能模塊,而GPS功能模塊的耗電量是巨大的。

在Android系統中,實現GPS定位的思路大致是:

1、獲取GPS的Location Provider。

2、將此Provider傳入到requestLocationUpdates()方法,讓Android系統獲知搜索位置方式。

3、創建實現瞭GpsStatus.Listener接口的對象,重寫onGpsStatusChanged()方法,向LocationManager添加次監聽器,檢測衛星狀態。(可選步驟)

根據以上思路,仿照Android定位功能(一)中的例子,可以很容易的得到以下實現代碼:(此代碼的實現前提是GPS功能模塊處於打開狀態)

 

public class MainActivity extends Activity {
    private LocationManager locationManager;
    private GpsStatus gpsstatus;
 
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
 
        // 獲取到LocationManager對象
        locationManager = (LocationManager) getSystemService(LOCATION_SERVICE);
 
        // 根據設置的Criteria對象,獲取最符合此標準的provider對象
        String currentProvider = locationManager.getProvider(
                LocationManager.GPS_PROVIDER).getName();
 
        // 根據當前provider對象獲取最後一次位置信息
        Location currentLocation = locationManager
                .getLastKnownLocation(currentProvider);
        // 如果位置信息為null,則請求更新位置信息
        if (currentLocation == null) {
            locationManager.requestLocationUpdates(currentProvider, 0, 0,
                    locationListener);
        }
        // 增加GPS狀態監聽器
        locationManager.addGpsStatusListener(gpsListener);
 
        // 直到獲得最後一次位置信息為止,如果未獲得最後一次位置信息,則顯示默認經緯度
        // 每隔10秒獲取一次位置信息
        while (true) {
            currentLocation = locationManager
                    .getLastKnownLocation(currentProvider);
            if (currentLocation != null) {
                Log.d("Location", "Latitude: " + currentLocation.getLatitude());
                Log.d("Location", "location: " + currentLocation.getLongitude());
                break;
            } else {
                Log.d("Location", "Latitude: " + 0);
                Log.d("Location", "location: " + 0);
            }
            try {
                Thread.sleep(10000);
            } catch (InterruptedException e) {
                Log.e("Location", e.getMessage());
            }
        }
    }
 
    private GpsStatus.Listener gpsListener = new GpsStatus.Listener() {
        // GPS狀態發生變化時觸發
        @Override
        public void onGpsStatusChanged(int event) {
            // 獲取當前狀態
            gpsstatus = locationManager.getGpsStatus(null);
            switch (event) {
            // 第一次定位時的事件
            case GpsStatus.GPS_EVENT_FIRST_FIX:
                break;
            // 開始定位的事件
            case GpsStatus.GPS_EVENT_STARTED:
                break;
            // 發送GPS衛星狀態事件
            case GpsStatus.GPS_EVENT_SATELLITE_STATUS:
                Toast.makeText(MainActivity.this,"GPS_EVENT_SATELLITE_STATUS",
                        Toast.LENGTH_SHORT).show();
                Iterable<GpsSatellite> allSatellites = gpsstatus
                        .getSatellites();
                Iterator<GpsSatellite> it = allSatellites.iterator();
                int count = 0;
                while (it.hasNext()) {
                    count++;
                }
                Toast.makeText(MainActivity.this, "Satellite Count:" + count,
                        Toast.LENGTH_SHORT).show();
                break;
            // 停止定位事件
            case GpsStatus.GPS_EVENT_STOPPED:
                Log.d("Location", "GPS_EVENT_STOPPED");
                break;
            }
        }
    };
 
    // 創建位置監聽器
    private LocationListener locationListener = new LocationListener() {
        // 位置發生改變時調用
        @Override
        public void onLocationChanged(Location location) {
            Log.d("Location", "onLocationChanged");
        }
 
        // provider失效時調用
        @Override
        public void onProviderDisabled(String provider) {
            Log.d("Location", "onProviderDisabled");
        }
 
        // provider啟用時調用
        @Override
        public void onProviderEnabled(String provider) {
            Log.d("Location", "onProviderEnabled");
        }
 
        // 狀態改變時調用
        @Override
        public void onStatusChanged(String provider, int status, Bundle extras) {
            Log.d("Location", "onStatusChanged");
        }
    };
}

 通過以上代碼中的註釋部分,可以清晰的知道Android定位功能裡相關方法的具體含義。希望對大傢有用。

另外,因為GPS的自身特性,此代碼在室內幾乎無法定位,所以建議再真正的實際項目裡,至少使用network和GPS兩種不同的Location Provider實現定位功能。

本人暫時未找到同時關閉網絡和GPS功能實現定位的方法,本人也未找到通過代碼在沒有ROOT的前提下直接代開網絡和GPS功能的代碼。如果大傢在這兩方面有自己的體會,請不吝賜教,留言評論或給出參考地址都可。大傢一同探討,一同進步。

 

 

發佈留言

發佈留言必須填寫的電子郵件地址不會公開。 必填欄位標示為 *