废话不多说,直接开始说说与实现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应该能够周期性的报告当前设备的位置信息。
这里通过一个代码示例,演示一下如何实现定位。
首先,在AndroidManifest.xml清单文件里需要加入ACCESS_FINE_LOCATION权限
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"></uses-permission>
其次,实现代码如下:
1 package com.test; 2 3 import java.io.IOException; 4 import java.util.List; 5 6 import android.app.Activity; 7 import android.location.Address; 8 import android.location.Criteria; 9 import android.location.Geocoder; 10 import android.location.Location; 11 import android.location.LocationListener; 12 import android.location.LocationManager; 13 import android.os.Bundle; 14 import android.util.Log; 15 import android.widget.Toast; 16 17 public class MainActivity extends Activity { 18 @Override 19 public void onCreate(Bundle savedInstanceState) { 20 super.onCreate(savedInstanceState); 21 setContentView(R.layout.main); 22 23 //获取到LocationManager对象 24 LocationManager locationManager = (LocationManager) getSystemService(LOCATION_SERVICE); 25 //创建一个Criteria对象 26 Criteria criteria = new Criteria(); 27 //设置粗略精确度 28 criteria.setAccuracy(Criteria.ACCURACY_COARSE); 29 //设置是否需要返回海拔信息 30 criteria.setAltitudeRequired(false); 31 //设置是否需要返回方位信息 32 criteria.setBearingRequired(false); 33 //设置是否允许付费服务 34 criteria.setCostAllowed(true); 35 //设置电量消耗等级 36 criteria.setPowerRequirement(Criteria.POWER_HIGH); 37 //设置是否需要返回速度信息 38 criteria.setSpeedRequired(false); 39 40 //根据设置的Criteria对象,获取最符合此标准的provider对象 41 String currentProvider = locationManager.getBestProvider(criteria, true); 42 Log.d("Location", "currentProvider: " + currentProvider); 43 //根据当前provider对象获取最后一次位置信息 44 Location currentLocation = locationManager.getLastKnownLocation(currentProvider); 45 //如果位置信息为null,则请求更新位置信息 46 if(currentLocation == null){ 47 locationManager.requestLocationUpdates(currentProvider, 0, 0, locationListener); 48 } 49 //直到获得最后一次位置信息为止,如果未获得最后一次位置信息,则显示默认经纬度 50 //每隔10秒获取一次位置信息 51 while(true){ 52 currentLocation = locationManager.getLastKnownLocation(currentProvider); 53 if(currentLocation != null){ 54 Log.d("Location", "Latitude: " + currentLocation.getLatitude()); 55 Log.d("Location", "location: " + currentLocation.getLongitude()); 56 break; 57 }else{ 58 Log.d("Location", "Latitude: " + 0); 59 Log.d("Location", "location: " + 0); 60 } 61 try { 62 Thread.sleep(10000); 63 } catch (InterruptedException e) { 64 Log.e("Location", e.getMessage()); 65 } 66 } 67 68 //解析地址并显示 69 Geocoder geoCoder = new Geocoder(this); 70 try { 71 int latitude = (int) currentLocation.getLatitude(); 72 int longitude = (int) currentLocation.getLongitude(); 73 List<Address> list = geoCoder.getFromLocation(latitude, longitude, 2); 74 for(int i=0; i<list.size(); i++){ 75 Address address = list.get(i); 76 Toast.makeText(MainActivity.this, address.getCountryName() + address.getAdminArea() + address.getFeatureName(), Toast.LENGTH_LONG).show(); 77 } 78 } catch (IOException e) { 79 Toast.makeText(MainActivity.this,e.getMessage(), Toast.LENGTH_LONG).show(); 80 } 81 82 } 83 84 //创建位置监听器 85 private LocationListener locationListener = new LocationListener(){ 86 //位置发生改变时调用 87 @Override 88 public void onLocationChanged(Location location) { 89 Log.d("Location", "onLocationChanged"); 90 Log.d("Location", "onLocationChanged Latitude" + location.getLatitude()); 91 Log.d("Location", "onLocationChanged location" + location.getLongitude()); 92 } 93 94 //provider失效时调用 95 @Override 96 public void onProviderDisabled(String provider) { 97 Log.d("Location", "onProviderDisabled"); 98 } 99 100 //provider启用时调用101 @Override102 public void onProviderEnabled(String provider) {103 Log.d("Location", "onProviderEnabled");104 }105 106 //状态改变时调用107 @Override108 public void onStatusChanged(String provider, int status, Bundle extras) {109 Log.d("Location", "onStatusChanged");110 }111 };112 }
由于代码里的Criteria对象对位置精度要求并不高,所以一般会返回“network”作为provider,而基于network的定位往往会存在一定的位置偏差,这对于需要精确定位的应用程序来说,显然不合要求。这时,需要则需要用到基于GPS的定位方法了。具体详情,请看后续博文 名字为:android定位实现(2)。