最近,我正在学习基于位置的服务,首先我们需要实现的是如何确定出自己所在的位置,这在Android中并不困难,主要用到LocationManager这个类就可以实现了。
我们首先了解一下LocationManager的基本用法,然后再通过一个例子来尝试获取一下自己当前的位置。
LocationManager的基本用法
使用LocationManager就必须要先获取到它的实例,我们可以调用Context的getSystemService()方法获取到,getSystemService()方法接收一个字符串参数用于确定获取系统的哪个服务,这里我们传入Context.LOCATION_SERVICE。因此,获取LocationManager的实例就可以写成:
LocationManager locationManager = (LocationManager)
getSystemService(Context.LOCATION_SERVICE);
接着我们需要选择一个位置提供器来确定设备当前的位置。Android中一般有三种位置提供器可供选择
- GPS_PROVIDER
- NETWORK_PROVIDER
- PASSIVE_PROVIDER
其中前两种使用的比较多,分别表示使用GPS定位和使用网络定位。这两种定位方式各有特点,GPS定位的精准度比较高,但是非常耗电,而网络定位的精准度稍差,但耗电量比较少。我们应该根据自己的实际情况来选择使用哪一种位置提供器,当位置精度要求非常高的时候,最好使用GPS_PROVIDER,而一般情况下,使用NETWORK_PROVIDER会更加得划算。
另外,定位功能必须要由用户主动去启用才行,不然任何应用程序都无法获取到手机当前的位置信息。进入手机的设置,其中第一个选项表示允许使用网络的方式来对手机进行定位,第二个选项表示允许使用GPS的方式来对手机进行定位
下面我们就来看一看,如何才能真正地开始定位操作。
将选择好的位置提供器传入到getLastKnownLocation()方法中,就可以得到一个Location对象,如下所示:
String provider = LocationManager.NETWORK_PROVIDER;
Location location=locationManager.getLastKnownLocation(provider);
这个Location对象中包含了经度、纬度、海拔等一系列的位置信息,然后从中取出我们所关心的那部分数据即可。
如果有些时候你想让定位的精度尽量高一些,但又不确定GPS定位的功能是否已经启用,这个时候就可以先判断一下有哪些位置提供器可用,如下所示:
List<String> providerList = locationManager.getProviders(true);
可以看到,getProviders()方法接收一个布尔型参数,传入true就表示只有启用的位置提供器才会被返回。之后再从providerList中判断是否包含GPS定位的功能就行了。
另外,那么我们怎样才能在设备位置发生改变的时候获取到最新的位置信息呢?LocationManager还提供了一个requestLocationUpdates()方法,只要传入一个LocationListener的实例,并简单配置几个参数就可以实现上述功能了,写法如下:
locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 5000, 10,
new LocationListener() {
@Override
public void onStatusChanged(String provider, int status, Bundle extras) {
}
@Override
public void onProviderEnabled(String provider) {
}
@Override
public void onProviderDisabled(String provider) {
}
@Override
public void onLocationChanged(Location location) {
}
});
这里requestLocationUpdates()方法接收四个参数,第一个参数是位置提供器的类型,第二个参数是监听位置变化的时间间隔,以毫秒为单位,第三个参数是监听位置变化的距离间隔,以米为单位,第四个参数则是LocationListener监听器。这样的话,LocationManager每隔5秒钟会检测一下位置的变化情况,当移动距离超过10米的时候,就会调用LocationListener的onLocationChanged()方法,并把新的位置信息作为参数传入。
好了,关于LocationManager的用法基本就是这么多,下面我们就通过一个例子来尝试一下吧。
确定自己位置的经纬度
新建一个LocationTest项目,修改activity_main.xml中的代码,如下所示:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent" >
<TextView
android:id="@+id/position_text_view"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
</LinearLayout>
布局文件中只有一个TextView控件,用于稍后显示设备位置的经纬度信息。
然后修改MainActivity中的代码,如下所示:
public class MainActivity extends Activity {
private TextView positionTextView;
private LocationManager locationManager;
private String provider;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
positionTextView = (TextView) findViewById(R.id.position_text_view);
locationManager = (LocationManager) getSystemService(Context. LOCATION_SERVICE);
// 获取所有可用的位置提供器
List<String> providerList = locationManager.getProviders(true);
if (providerList.contains(LocationManager.GPS_PROVIDER)) {
provider = LocationManager.GPS_PROVIDER;
} else if (providerList.contains(LocationManager.NETWORK_PROVIDER)) {
provider = LocationManager.NETWORK_PROVIDER;
} else {
// 当没有可用的位置提供器时,弹出Toast提示用户
Toast.makeText(this, "No location provider to use", Toast.LENGTH_SHORT).show();
return;
}
Location location = locationManager.getLastKnownLocation(provider);
if (location != null) {
// 显示当前设备的位置信息
showLocation(location);
}
locationManager.requestLocationUpdates(provider, 5000, 1, locationListener);
}
protected void onDestroy() {
super.onDestroy();
if (locationManager != null) {
// 关闭程序时将监听器移除
locationManager.removeUpdates(locationListener);
}
}
LocationListener locationListener = new LocationListener() {
@Override
public void onStatusChanged(String provider, int status, Bundle extras) {
}
@Override
public void onProviderEnabled(String provider) {
}
@Override
public void onProviderDisabled(String provider) {
}
@Override
public void onLocationChanged(Location location) {
// 更新当前设备的位置信息
showLocation(location);
}
};
private void showLocation(Location location) {
String currentPosition = "latitude is " + location.getLatitude() + "\n"
+ "longitude is " + location.getLongitude();
positionTextView.setText(currentPosition);
}
}
在onCreate()方法中首先是获取到了LocationManager的实例,然后调用getProviders()方法去得到所有可用的位置提供器,接下来再调用getLastKnownLocation()方法就可以获取到记录当前位置信息的Location对象了,这里我们将Location对象传入到showLocation()方法中,经度和纬度的值就会显示到TextView上了。然后为了要能监测到位置信息的变化,下面又调用了requestLocationUpdates()方法来添加一个位置监听器,设置时间间隔是5秒,距离间隔是1米,并在onLocationChanged()方法中时时更新TextView上显示的经纬度信息。最后当程序关闭时,我们还需要调用removeUpdates()方法来将位置监听器移除,以保证不会继续耗费手机的电量。
另外,获取设备当前的位置信息也是要声明权限的,因此还需要修改AndroidManifest.xml中的代码,如下所示:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.locationtest">
……
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
……
</manifest>
现在运行一下程序,发现代码中有报错的地方
百度了下,大意是:
调用需要的权限可能被用户拒绝:代码应该显式地检查权限是否可用(用checkPermission)或显式地处理一个潜在的“SecurityException”
所以,修改代码如下:
if(ContextCompat.checkSelfPermission(this,android.Manifest.permission.ACCESS_FINE_LOCATION) == PackageManager.PERMISSION_GRANTED) {
//此处的判定是主要问题,API23之后需要先判断之后才能调locationManager中的方法
if (providerList.contains(LocationManager.GPS_PROVIDER)) {
provider = LocationManager.GPS_PROVIDER;
} else if (providerList.contains(LocationManager.NETWORK_PROVIDER)) {
provider = LocationManager.NETWORK_PROVIDER;
} else {
// 当没有可用的位置提供器时,弹出Toast提示用户
Toast.makeText(this, "No location provider to use", Toast.LENGTH_SHORT).show();
return;
}
Location location = locationManager.getLastKnownLocation(provider);
if (location != null) {
// 显示当前设备的位置信息
showLocation(location);
}
locationManager.requestLocationUpdates(provider, 5000, 1, locationListener);
}
}
protected void onDestroy() {
super.onDestroy();
if (ContextCompat.checkSelfPermission(this, android.Manifest.permission.ACCESS_FINE_LOCATION) ==PackageManager.PERMISSION_GRANTED) {
//此处同样需要判定之后才能调用removeUpdates方法
if (locationManager != null) {
// 关闭程序时将监听器移除
locationManager.removeUpdates(locationListener);
}
}
}
这样代码报错就解决了
另外需要注意,建议代码都在手机上运行,在手机上得到真实的位置数据,之后如果你拿着手机随处移动,就可以看到界面上的经纬度信息是会变化的。
作者:章露颖:原文地址