基于位置的服务包括三个方面:获取位置、地图服务、地理编码服务。
获取位置是所有基于位置的服务的基础,Android获取位置使用LocationManager类。首先获取LocationManager的实例,然后一般要检查位置服务是否开启。如果未开启,就打开设置位置服务界面。如果已开启,再检查应用的定位权限是否允许。都通过了,就可以设置一个位置监听器,有了位置信息就会调用监听器的相应方法。设置监听器时可以设置位置监听的最小时间间隔和最小距离间隔,只有这两个条件都满足时才会有位置信息。流程和关键代码如下图:
下面是一个获取位置的例子。例子的界面如下,最上面一行是两项设置,记录位置的最小时间间隔和最小距离,只有这两个条件同时满足,才会产生一个位置数据。第二行是两个按钮,分别启动和停止位置监听。再下面是一个文本,显示监听到的位置信息。
例子中,首先要在onCreate中获取LocationManager实例,并检查设备是否开启了位置服务,代码如下:
manager = (LocationManager)getSystemService(Context.LOCATION_SERVICE);
if(!manager.isProviderEnabled(LocationManager.GPS_PROVIDER) &&
!manager.isProviderEnabled(LocationManager.NETWORK_PROVIDER)) {
askLocationSettings();
}
可以从手机最上面的功能栏中开启/关闭手机的位置服务,如下图:
如果位置服务未开启,可以询问用户是否转到位置服务设置界面。位置服务设置界面通过Intent来启动,动作是android.provider.Settings.ACTION_LOCATION_SOURCE_SETTINGS。代码如下:
void askLocationSettings(){
AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setTitle("开启位置服务");
builder.setMessage("本应用需要开启位置服务,是否去设置界面开启位置服务?");
builder.setPositiveButton("是", new DialogInterface.OnClickListener() {
public void onClick(DialogInterface arg0, int arg1) {
Intent intent = new Intent(
android.provider.Settings.ACTION_LOCATION_SOURCE_SETTINGS);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
MainActivity.this.startActivity(intent);
}
});
builder.setNegativeButton("否", new DialogInterface.OnClickListener() {
public void onClick(DialogInterface arg0, int arg1) {
Toast.makeText(MainActivity.this, "No location provider to use",
Toast.LENGTH_SHORT).show();
}
});
builder.show();
}
位置服务设置界面如下:
如果设备开启了位置服务,就可以创建一个位置信息监听器,在监听时需要传给LocationManager。在事件监听方法中将经度、纬度、高度、速度等位置信息输出。代码如下:
listener = new LocationListener(){
public void onLocationChanged(Location arg0) {
String txt = "Longitude:" + arg0.getLongitude()
+ ", Latitude:" + arg0.getLatitude()
+ ", Altitude" + arg0.getAltitude()
+ ", Speed:" + arg0.getSpeed();
// 经度,纬度,高度,速度
tvResult.append(txt + "\n");
}
public void onProviderDisabled(String arg0) {}
public void onProviderEnabled(String arg0) {}
public void onStatusChanged(String arg0, int arg1, Bundle arg2) {}
};
启动位置监听需要调用LocationManager的requestLocationUpdates()方法,方法的第一个参数是位置服务提供者。一个设备上可能有多个位置服务提供者,比如gps,wifi,北斗等,所以要先根据定位需求寻找一个最匹配的提供者,使用的方法是LocationManager的getBestProvider()方法。另外,获取当前位置需要一定的时间,而有些应用又已启动就需要一个定位信息,此时可以使用上次使用时的最后已知位置。该位置保存在缓存中,可以通过LocationManager的getLastKnownLocation()方法获得。具体代码如下:
void startListening(){
Criteria crt = new Criteria(); // 位置监听标准
crt.setHorizontalAccuracy(Criteria.ACCURACY_HIGH); // 水平精度高
crt.setAltitudeRequired(true); // 需要高度
String provider = manager.getBestProvider(crt, true); // 寻找最匹配的provider
Location l = manager.getLastKnownLocation(provider); // 取最后已知位置,即缓存中的位置
if(l!=null) tvResult.append(provider + "-LastKnown:" + l.toString() + "\n");
long period = Long.parseLong(etPeriod.getText().toString()); // 最小时间间隔
int distance = Integer.parseInt(etDistance.getText().toString()); // 最小距离
manager.requestLocationUpdates(provider, period*1000, distance, listener); // 开始监听
tvResult.append(provider + "-Location listener started.\n");
}
Android6.0以上系统需要应用运行时进行动态权限申请,所以在开始监听位置前需要检查权限,如果没有许可就进行询问。主要代码是:
if (ActivityCompat.checkSelfPermission(activity, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
ActivityCompat.requestPermissions(activity, new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, 101);
return;
}
停止位置监听使用LocationManager的removeUpdates()方法,代码如下:
manager.removeUpdates(listener);
tvResult.append("Location listener stoped.\n");
监听位置需要在配置文件里声明权限:
例子中用到的相关类包括:
1)android.location.LocationManager类,方法有:
2)android.location.Criteria类,方法有:
3)android.location.Location类,方法有:
4)android.location.LocationListener接口,方法有: