Android 实现模拟地图定位功能
实现原理:
??手机定位方式目前有4种:基站定位,WIFI定位,GPS定位,AGPS定位。
??本工程利用手机自带的"模拟位置"功能实现运行时修改LocationManager结果。
??原理:使用android自带的调试api,模拟gps provider的结果。 ??Android 6.0系统以下,可以通过Setting.Secure.ALLOW_MOCK_LOCATION获取是否【允许模拟位置】,当【允许模拟位置】开启时,可addTestProvider;
??Android 6.0系统及以上,弃用Setting.Secure.ALLOW_MOCK_LOCATION变量,没有【允许模拟位置】选项,
增加【选择模拟位置信息应用】,此时需要选择当前应用,才可以addTestProvider,
但未找到获取当前选择应用的方法,因此通过addTestProvider是否成功来判断是否可用模拟位置。
代码分析:
MockLocationManager:模拟地址管理类
??首先通过Android系统模拟位置管理器LocationManager获取系统模拟位置服务,Android 6.0以下,通过Setting.Secure.ALLOW_MOCK_LOCATION判断是否可模拟位置,Android 6.0及以上,需要【选择模拟位置信息应用】,未找到方法,因此通过addTestProvider是否可用判断。
/**
* 模拟位置是否启用
* 若启用,则addTestProvider
*/
public boolean getUseMockPosition(Context context) {
// Android 6.0以下,通过Setting.Secure.ALLOW_MOCK_LOCATION判断
// Android 6.0及以上,需要【选择模拟位置信息应用】,未找到方法,因此通过addTestProvider是否可用判断
boolean canMockPosition = (Settings.Secure.getInt(context.getContentResolver(), Settings.Secure.ALLOW_MOCK_LOCATION, 0) != 0)
|| Build.VERSION.SDK_INT > 22;
if (canMockPosition && hasAddTestProvider == false) {
try {
for (String providerStr : mockProviders) {
LocationProvider provider = locationManager.getProvider(providerStr);
if (provider != null) {
locationManager.addTestProvider(
provider.getName()
, provider.requiresNetwork()
, provider.requiresSatellite()
, provider.requiresCell()
, provider.hasMonetaryCost()
, provider.supportsAltitude()
, provider.supportsSpeed()
, provider.supportsBearing()
, provider.getPowerRequirement()
, provider.getAccuracy());
} else {
if (providerStr.equals(LocationManager.GPS_PROVIDER)) {
locationManager.addTestProvider(
providerStr
, true, true, false, false, true, true, true
, Criteria.POWER_HIGH, Criteria.ACCURACY_FINE);
} else if (providerStr.equals(LocationManager.NETWORK_PROVIDER)) {
locationManager.addTestProvider(
providerStr
, true, false, true, false, false, false, false
, Criteria.POWER_LOW, Criteria.ACCURACY_FINE);
} else {
locationManager.addTestProvider(
providerStr
, false, false, false, false, true, true, true
, Criteria.POWER_LOW, Criteria.ACCURACY_FINE);
}
}
locationManager.setTestProviderEnabled(providerStr, true);
locationManager.setTestProviderStatus(providerStr, LocationProvider.AVAILABLE, null, System.currentTimeMillis());
}
hasAddTestProvider = true;
// 模拟位置可用
canMockPosition = true;
} catch (SecurityException e) {
canMockPosition = false;
}
}
if (canMockPosition == false) {
stopMockLocation();
}
return canMockPosition;
}
```http://www.biyezuopin.vip
接下来设置模拟经纬度数据:
```java
// 模拟位置(addTestProvider成功的前提下)
for (String providerStr : mockProviders) {
Location mockLocation = new Location(providerStr);
mockLocation.setLatitude(latitude);
// 维度(度)
mockLocation.setLongitude(longitude);
// 经度(度)
mockLocation.setAccuracy(0.1f);
// 精度(米)
mockLocation.setTime(new Date().getTime());
// 本地时间
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
mockLocation.setElapsedRealtimeNanos(SystemClock.elapsedRealtimeNanos());
}
locationManager.setTestProviderLocation(providerStr, mockLocation);
}
```http://www.biyezuopin.vip
取消模拟定位方法:
```java
/**
* 取消位置模拟,以免启用模拟数据后无法还原使用系统位置
* 若模拟位置未开启,则removeTestProvider将会抛出异常;
* 若已addTestProvider后,关闭模拟位置,未removeTestProvider将导致系统GPS无数据更新;
*/
public void stopMockLocation() {
if (hasAddTestProvider) {
for (String provider : mockProviders) {
try {
locationManager.removeTestProvider(provider);
} catch (Exception ex) {
// 此处不需要输出日志,若未成功addTestProvider,则必然会出错
// 这里是对于非正常情况的预防措施
}
}
hasAddTestProvider = false;
}
}
??注册位置服务,获取系统位置
// 注册位置服务,获取系统位置
if (ActivityCompat.checkSelfPermission(context, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED && ActivityCompat.checkSelfPermission(context, Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
//ActivityCompat#requestPermissions
// here to request the missing permissions, and then overriding
//public void onRequestPermissionsResult(int requestCode, String[] permissions,
//int[] grantResults)
// to handle the case where the user grants the permission. See the documentation
// for ActivityCompat#requestPermissions for more details.
return;
}
mockLocationManager.locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0, locationListener);
??最后通过LocationListener.onLocationChanged()回调方法获取GPS定位数据:
private LocationListener locationListener = new LocationListener() {
@Override
public void onLocationChanged(final Location location) {
setLocationData(location);
}@Override
public void onStatusChanged(String provider, int status, Bundle extras) {}@Override
public void onProviderEnabled(String provider) {}@Override
public void onProviderDisabled(String provider) {}
};
/**
* 获取到模拟定位信息,并显示
*
* @param location 定位信息
*/
private void setLocationData(Location location) {
tvProvider.setText(location.getProvider());
tvTime.setText(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date(location.getTime())));
tvLatitude.setText(location.getLatitude() + " °");
tvLongitude.setText(location.getLongitude() + " °");
}
使用模拟定位需先开启系统设置中的模拟位置:
- Android 6.0 以下:【开发者选项 -> 允许模拟位置】
文章图片
- Android 6.0 及以上:【开发者选项 -> 选择模拟位置信息应用】
文章图片
【android|Android 实现模拟地图定位功能】参考链接:
- 1、【科普】GPS、Wifi等各种手机定位方式的含义及原理详解
- 2、Android 使用模拟位置(支持Android 6.0)
- 3、http://www.biyezuopin.vip
推荐阅读
- 2022年【米哈游】 金三银四 三月内推开始啦!不加班福利好,200+个岗位任你挑选,赶快来看吧!
- 架构师|年后腾讯、阿里、滴滴后台面试题汇总总结 — (含答案)
- 如何写一份优秀的Java程序员简历
- 多线程控制 countDownLatch、CyclicBarrier、Semaphore 总结
- 编程语言|TypeScript的另一面(类型编程)
- 热度不再,但各大厂仍对中高级Android开发趋之若鹜......
- 笔记|栈和队列常见oj题