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应该能够周期性的报告当前设备的位置信息。
这里通过一个代码示例,演示一下如何实现定位。
首先,在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 publicclass MainActivity extends Activity {
18 @Override
19 publicvoid 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 publicvoid 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 publicvoid onProviderDisabled(String provider) {
97 Log.d("Location", "onProviderDisabled");
98 }
99
100 //provider启用时调用
101 @Override
102 publicvoid onProviderEnabled(String provider) {
103 Log.d("Location", "onProviderEnabled");
104 }
105
106 //状态改变时调用
107 @Override
108 publicvoid onStatusChanged(String provider, int status, Bundle extras) {
109 Log.d("Location", "onStatusChanged");
110 }
111 };
112 }
由于代码里的Criteria对象对位置精度要求并不高,所以一般会返回“network”作为provider,而基于network的定位往往会存在一定的位置偏差,这对于需要精确定位的应用程序来说,显然不合要求。这时,需要则需要用到基于GPS的定位方法了