前言:
我们知道我们的APP有可能需要获取一些地理位置信息。比如定位用户当前的位置,自动选定城市或者区域等。所以这次做个关于定位的一些总结。
正文
我们按照二大块来进行分析:一块是相关权限,一块是具体获取地理信息的相关代码。(而实际开发代码中,代码这二块是写在一起的,单纯是为了文章分析从而分开。)
1.相关权限
这里的权限我特指了二块:
- 一个是本身我们平常开发的app需要获取各种权限,比如相机等,这时候我们既然要获取当前手机的地理信息,肯定也要有一个Location相关的权限。
- 本身手机需要打开相应的定位功能,不然app有权限获取,但是手机关闭了整个的定位功能,就还是获取不到。
1.1 app获取手机权限
emmm……这块我觉得应该不需要花更多的时间来说明了吧,主要就是:
- 检查权限 (checkSelfPermission)
- 请求权限(requestPermissions)
- 回调事件处理(onRequestPermissionsResult) 而我们要申请的权限无非就是Location相关的权限。
android.permission.ACCESS_COARSE_LOCATION
允许一个程序访问CellID或WiFi热点来获取粗略的位置
android.permission.ACCESS_FINE_LOCATION
允许一个程序访问精良位置(如GPS)
我们可以看到第一个权限中的英文单词COARSE
是粗略
的意思,所以在想要粗略的获取一个地理位置的时候,比如我们通过网络来获取,我们只需要申请这个权限即可;第二个权限中的英文单词FINE
说明是精确度高
的,比如我们需要通过GPS来获取权限的时候,我们就需要申请这个权限。
一般来说我们的app这二个权限都会申请,因为会需要GPS配合网络一起来确定地理位置信息。
1.2 手机的定位开关
在确定我们的app本身已经具有了定位权限后,我们需要知道本身的手机是否已经打开了定位功能。
public static boolean isLocServiceEnable(Context context) {
LocationManager locationManager = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE);
boolean gps = locationManager.isProviderEnabled(LocationManager.GPS_PROVIDER);
boolean network = locationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER);
if (gps || network) {
return true;
}
return false;
}
我们可以看到上面我们提过一般来获取定位是靠GPS和NetWork二种(为啥是一般呢,因为还有一种 PASSIVE,后面会讲到)。所以我们需要判定这二个功能是否可用。(如果用户把定位功能给关了,那肯定二个都返回false。)
那这时候假如我们发现用户把定位功能关了。我们肯定需要提示用户,然后协助用户跳到该设置界面,从而让用户把定位功能打开 (毕竟一般的普通用户,可能还真的让他去设置界面找,一时半会还真找不到,毕竟安卓机型太多,每个地方都不同 )。
比如我们弹出一个弹框,提示用户,按确定按钮的时候跳转到设置的定位界面:
AlertDialog.Builder builder = new AlertDialog.Builder(activity);
builder.setMessage("尚未开启位置定位服务");
builder.setPositiveButton("开启", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
//启动定位Activity
Intent intent = new Intent(Settings.ACTION_LOCATION_SOURCE_SETTINGS);
//比如我们这里设定requestCode为 1
activity.startActivityForResult(intent , 1);
}
});
builder.setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
}
});
我们可以看到通过隐式启动Action为Settings.ACTION_LOCATION_SOURCE_SETTINGS
即可,但是这里记得用使用startActivityForResult
而不是startActivity
,我看很多网上的的写法是用startActivity
,单纯跳转过去是没问题,但是我们需要知道返回的结果,万一用户跳转过去后没有打开呢。
既然我们用了statActivityForResult
来启动,当我们返回回到自己的app界面的时候,在onActivityForResult
中需要来判断,本来因为习惯性思维,所以以为自动在onActivityForResult
的返回参数resultCode
可以用来判断,后来发现不管开启不开启,都是返回RESULT_CANCELED
,也就是0,毕竟在那个设置界面我们并没有设定setResult(xxx)
;所以当判断了requestCode
之后,我们需要重新判断一次定位是否可用了。
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == 1) {
//我们通过上面提过的方法,再次去判断是否gps和network的provider都无效。
if (!isLocServiceEnable(MainActivity.this)) {
Toast.makeText(this, "未开启定位功能,请手动选择地址位置", Toast.LENGTH_LONG).show();
} else {
//去获取具体的地理位置信息...
}
}
}
2 获取地理位置
我们上面提到了我们想要获取地理位置的时候,需要具备上面的基本权限,然后才能正常使用我们的相关api去获取信息。
主要是通过“`LocationManager“这个类。
但是android.location包下的并不是谷歌推荐的:
翻译过来就是:此API不是访问Android位置的推荐方法。 Google位置服务API是Google Play服务的一部分,是向您的应用添加位置感知功能的首选方式。 它提供了更简单的API,更高的精度,低功耗的地理围栏等等。 如果您当前正在使用android.location API,强烈建议您尽快切换到Google Location Services API。
而是推荐the Google Location Services API ,然后你懂得….emmm……..
2.1 直接获取地理信息
使用getLastKnownLocation
方法获取:
//获取LocationManager的实例对象
locationManager = (LocationManager) activity.getSystemService(Context.LOCATION_SERVICE);
//获取支持的provider列表
List<String> providers = locationManager.getProviders(true);
Location bestLocation = null;
//遍历provider列表
for (String provider : providers) {
//通过getLastKnowLocation方法来获取
Location l = locationManager.getLastKnownLocation(provider);
if (l == null) {
continue;
}
if (bestLocation == null || l.getAccuracy() < bestLocation.getAccuracy()) {
// Found best last known location: %s", l);
bestLocation = l;
}
}
这里需要注意的是,为啥通过循环provider来获取,比如有些人会问,我开启了GPS,我想通过GPS来定位,我不是直接getLastKnowLocation(LocationManager.GPS_PROVIDER)
就可以了吗?理论上是没问题的,但是大部分时候获取到的都是null , 毕竟GPS本身定位时间也会很久,而且如果在室内就更加GG了。
所以网上经常看到有提问:
当然解决方式也有很多,有些人直接通过while循环,比如一直请求:
while(xxx = null){
xxx = getLastKnowLocation(LocationManager.GPS_PROVIDER);
}
这还不算坑爹,我用了华为和小米手机,小米手机使用这个GPS来获取Location,一下子就获取了。华为我写了while循环,等了很久很久,也还是一直是null。(居然还跟不同牌子手机都有关系)
所以最终我是遍历了provider来获取最佳的地址来解决的,如果获取不到GPS定位,也会有network辅助。
也可以参考相关的链接了解一下:Android 成功 使用GPS获取当前地理位置(解决getLastKnownLocation 返回 null),不过貌似也没有找到百分百直接获取GPS定位获取信息的方式。
2.2 监视位置变化
使用requestLocationUpdates方法来获取。
public void requestLocationUpdates( String provider, long minTime, float minDistance, LocationListener listener) {
}
我们可以看到传入provider,最小更新时间,最小的更新距离,然后就是回调listener。
所以我们重点在于LocationListener
:
mLocationListener = new LocationListener() {
@Override
public void onLocationChanged(Location location) {
//比如判断location是否为null,然后根据Location来转换成相关的地址位置信息。
}
@Override
public void onStatusChanged(String provider, int status, Bundle extras) {
}
@Override
public void onProviderEnabled(String provider) {
}
@Override
public void onProviderDisabled(String provider) {
//比如在provider失效了就取消监听
locationManager.removeUpdates(this);
}
};
切记,在某个你需要的条件下,通过removeUpdates()去取消监听,比如你可能在onPause中去取消等。
我们在onLocationChanged
方法中获取到了Location
对象,就可以去获取相关信息了。
- 通过Location来获取相关的经纬度:
double latitude = location.getLatitude();
double longitude = location.getLongitude();
- 通过Geocoder来把经纬度转换成相应的Address集合:
Geocoder geocoder = new Geocoder(context);
List<Address> addresses = geocoder.getFromLocation(latitude, longitude, 1);
- 最后通过Address对象中的相关属性,拼接出自己想要的相关信息。
address.getCountryName() //国家
address.getPostalCode() //邮编
address.getCountryCode() //国家编码
address.getAdminArea() //省份
address.getSubAdminArea() //二级省份
address.getThoroughfare() //道路
address.getSubLocality() //二级城市
.......
.......
结语:
emm…….大家轻喷即可。。。。
今天的文章项目需求讨论 – 定位功能小结分享到此就结束了,感谢您的阅读。
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
如需转载请保留出处:https://bianchenghao.cn/16411.html