使用Android提供的模拟任意地理位置,报java.lang.IllegalArgumentException: Incomplete location object异常的解决方法

本文介绍了解决在Android中模拟地理位置时出现的IllegalArgumentException异常的方法。通过完善Location对象属性,包括设置时间戳和精度,或者使用反射调用makeComplete()方法来解决此问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

使用Android提供的模拟任意地理位置的接口,执行如下方法时,会报异常:

[java]  view plain  copy
 print ?
  1. Location localLocation = getLoc(LocationManager.GPS_PROVIDER);  
  2. mLocationManager.setTestProviderLocation(LocationManager.GPS_PROVIDER,  
  3.         localLocation);  

 

[plain]  view plain  copy
 print ?
  1. 12-10 14:54:24.497: W/System.err(27548): java.lang.IllegalArgumentException: Incomplete location object, missing timestamp or accuracy? Location[gps 22.546054,114.025974 acc=0 et=?!? alt=0.0 vel=0.0 bear=0.0]  
  2. 12-10 14:54:24.497: W/System.err(27548):  at android.location.LocationManager.setTestProviderLocation(LocationManager.java:1218)  
  3. 12-10 14:54:24.497: W/System.err(27548):  at com.zhao3546.MockLocationServicePro$1.run(MockLocationServicePro.java:55)  
  4. 12-10 14:54:24.497: W/System.err(27548):  at android.os.Handler.handleCallback(Handler.java:725)  
  5. 12-10 14:54:24.497: W/System.err(27548):  at android.os.Handler.dispatchMessage(Handler.java:92)  
  6. 12-10 14:54:24.497: W/System.err(27548):  at android.os.Looper.loop(Looper.java:137)  
  7. 12-10 14:54:24.497: W/System.err(27548):  at android.app.ActivityThread.main(ActivityThread.java:5106)  
  8. 12-10 14:54:24.497: W/System.err(27548):  at java.lang.reflect.Method.invokeNative(Native Method)  
  9. 12-10 14:54:24.497: W/System.err(27548):  at java.lang.reflect.Method.invoke(Method.java:511)  
  10. 12-10 14:54:24.497: W/System.err(27548):  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:898)  
  11. 12-10 14:54:24.500: W/System.err(27548):  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:665)  
  12. 12-10 14:54:24.500: W/System.err(27548):  at dalvik.system.NativeStart.main(Native Method)  

 

这个问题可以通过两个方法解决。

方法一:

主要是这句,location.setElapsedRealtimeNanos(SystemClock.elapsedRealtimeNanos());

[java]  view plain  copy
 print ?
  1. @SuppressLint("NewApi")  
  2. private Location getLoc(String provider)  
  3. {  
  4.     Location location = new Location(provider);  
  5.     location.setLatitude(lat);  
  6.     location.setLongitude(lng);  
  7.     location.setAltitude(altitude);  
  8.     location.setBearing(bearing);  
  9.     location.setSpeed(speed);  
  10.     location.setAccuracy(accuracy);  
  11.     location.setTime(System.currentTimeMillis());  
  12.     location.setElapsedRealtimeNanos(SystemClock.elapsedRealtimeNanos());  
  13.     return location;  
  14. }  


方法二,加上下面的反射调用:

[java]  view plain  copy
 print ?
  1. try  
  2. {  
  3.     Method method = Location.class.getMethod("makeComplete");  
  4.     if (method != null)  
  5.     {  
  6.         method.invoke(localLocation);  
  7.     }  
  8. }  
  9. catch (NoSuchMethodException e)  
  10. {  
  11.     e.printStackTrace();  
  12. }  
  13. catch (Exception e)  
  14. {  
  15.     e.printStackTrace();  
  16. }  

 

第二个方法调用的 makeComplete() 方法在Android源码中的实现如下,其实和第一个方法异曲同工,还是第一个方法更优。

[java]  view plain  copy
 print ?
  1. /** 
  2.  * Helper to fill incomplete fields. 
  3.  * 
  4.  * <p>Used to assist in backwards compatibility with 
  5.  * Location objects received from applications. 
  6.  * 
  7.  * @see #isComplete 
  8.  * @hide 
  9.  */  
  10. public void makeComplete() {  
  11.     if (mProvider == null) mProvider = "?";  
  12.     if (!mHasAccuracy) {  
  13.         mHasAccuracy = true;  
  14.         mAccuracy = 100.0f;  
  15.     }  
  16.     if (mTime == 0) mTime = System.currentTimeMillis();  
  17.     if (mElapsedRealtimeNanos == 0) mElapsedRealtimeNanos = SystemClock.elapsedRealtimeNanos();  
  18. }  


### 解决方案 `java.lang.IllegalArgumentException: Unknown package` 错误通常发生在尝试通过 `PackageManager` 查询不存在的包名或者未正确定义目标应用程序的情况下。以下是针对此问题的具体分析和解决方案: #### 1. **确认目标包是否存在** 确保目标应用程序已经安装到设备上,并且其包名正确无误。可以通过以下方式验证: - 使用命令行工具 `adb shell pm list packages | grep com.example.targetapp` 来检查目标包是否存在于当前设备中[^1]。 - 如果目标包确实不存在,则需要重新安装该应用。 #### 2. **在 `<queries>` 中声明目标包** 自 Android 11 (API Level 30) 起,为了增强隐私保护,只有显式声明的目标包才能被查询或访问。因此,在应用的 `AndroidManifest.xml` 文件中添加 `<queries>` 元素并指定目标包名是必要的。例如: ```xml <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.myapp"> <queries> <!-- 声明要查询的应用程序 --> <package android:name="com.example.targetapp" /> </queries> </manifest> ``` 如果没有正确配置 `<queries>` 部分,可能会触发 `IllegalArgumentException: Unknown package` 的异常[^2]。 #### 3. **动态检测包的存在性** 即使已经在 manifest 文件中声明了目标包,仍需在运行时进一步验证目标包是否可用。可以使用以下代码来实现这一点: ```java public boolean isPackageInstalled(String packageName, PackageManager packageManager) { try { packageManager.getApplicationInfo(packageName, PackageManager.GET_META_DATA); return true; } catch (PackageManager.NameNotFoundException e) { return false; } } ``` 调用上述函数前,请先获取系统的 `PackageManager` 实例[^3]: ```java PackageManager packageManager = getPackageManager(); if (!isPackageInstalled("com.example.targetapp", packageManager)) { throw new IllegalArgumentException("Unknown package"); } ``` #### 4. **调试与日志记录** 当遇到此类错误时,建议增加详细的日志输出以便于定位问题所在。例如: ```java Log.e("PACKAGE_ERROR", "Failed to find package: " + packageName, exception); ``` --- ### 总结 综上所述,解决 `java.lang.IllegalArgumentException: Unknown package` 的核心在于两个方面:一是确保目标包的有效性和可访问性;二是遵循最新的 Android 安全策略调整应用配置。具体操作包括但不限于核实包名准确性、更新 Manifest 文件以及执行额外的安全校验逻辑。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值