基于位置的服务之如何找到自己的位置

本文介绍了如何在Android应用中实现基于位置的服务。通过LocationManager类获取设备位置信息,并使用不同位置提供器如GPS_PROVIDER和NETWORK_PROVIDER。同时展示了如何通过监听器实时获取位置更新。

最近,我正在学习基于位置的服务,首先我们需要实现的是如何确定出自己所在的位置,这在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);
                        }
                    }
                }

这样代码报错就解决了


另外需要注意,建议代码都在手机上运行,在手机上得到真实的位置数据,之后如果你拿着手机随处移动,就可以看到界面上的经纬度信息是会变化的。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值