我有一个带有按钮的简单活动,它使用 LocationManager 来尝试获取当前位置:
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Button buttonGps = findViewById(R.id.buttonGps);
final Context activity = this;
buttonGps.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (!checkForPhonePermission()) {
return;
}
LocationManager locationManager = (LocationManager) activity.getSystemService(Context.LOCATION_SERVICE);
locationManager.requestSingleUpdate(LocationManager.GPS_PROVIDER, new LocationListener() {
@Override
public void onLocationChanged(@NonNull Location location) {
Log.d(TAG, location.getLatitude() + "");
}
}, Looper.myLooper());
}
});
}
我在 Android Studio 中创建了一个 API 级别为 22 的模拟器,在授予权限后,在模拟器的 GPS 开启的情况下,单击按钮时应用程序崩溃并出现此错误:
java.lang.AbstractMethodError: abstract method "void android.location.LocationListener.onStatusChanged(java.lang.String, int, android.os.Bundle)"
如果我在 GPS 关闭的情况下按下按钮,则会收到此错误:
java.lang.AbstractMethodError: abstract method "void android.location.LocationListener.onProviderDisabled(java.lang.String)"
如果我在小米 A1 上尝试该应用程序,它只会在 GPS 关闭并单击按钮时崩溃,而当 GPS 打开并按下按钮时它不会崩溃。
从文档来看,这些方法被标记为默认,所以我不需要实现它们。
这种行为有什么原因吗?
只需在代码末尾添加这三个函数即可:
@Override
public void onProviderEnabled(@NonNull String provider) {
}
@Override
public void onProviderDisabled(@NonNull String provider) {
}
@Override
public void onStatusChanged(String provider, int status, Bundle extras) {
}
并编写您想要在状态更改时执行的代码
我对上面的答案并不完全满意,决定详细说明一下。
正如 @sunlover3 正确指出的那样,有问题的方法的实现已更改。在 API 30 之前,它们是抽象的,这意味着如果compileSdk 不是 30,OP 的代码将给出编译错误(这些方法必须已实现)。但在API 30中,他们添加了空的默认方法的实现,这意味着如果compileSdk是30,编译器检查这些方法并且没有给出错误,则它们已经实现了。
但是,如果在版本低于 30 的旧操作系统上运行带有compileSdk 30 的应用程序,这些方法的实现仍然是旧的,i.o.没有,它们是抽象的!因此,出现错误。
老实说,这是一个明显的 SDK 开发错误,这样的事情绝对不能发生。最坏的情况应该是出现编译时警告,表明仍应重写这些方法以实现旧版本的兼容性。我希望在这个话题上比我不那么懒的人能够向 google tracker 报告这个问题:)
这是我的情况。在Android 9(API 28)和Android 10(API 29)上,在android.location.LocationManager类中有这个方法:
private void _handleMessage(Message msg) {
switch (msg.what) {
case TYPE_LOCATION_CHANGED:
Location location = new Location((Location) msg.obj);
mListener.onLocationChanged(location);
break;
case TYPE_STATUS_CHANGED:
Bundle b = (Bundle) msg.obj;
String provider = b.getString("provider");
int status = b.getInt("status");
Bundle extras = b.getBundle("extras");
mListener.onStatusChanged(provider, status, extras);
break;
case TYPE_PROVIDER_ENABLED:
mListener.onProviderEnabled((String) msg.obj);
break;
case TYPE_PROVIDER_DISABLED:
mListener.onProviderDisabled((String) msg.obj);
break;
}
locationCallbackFinished();
}
这似乎是这样称呼的。另外,Android 29 和 Android 30 之间 LocationListener 的实现差异是这样的:
Android 29:
public interface LocationListener {
void onLocationChanged(Location location);
@Deprecated
void onStatusChanged(String provider, int status, Bundle extras);
void onProviderEnabled(String provider);
void onProviderDisabled(String provider);
}
Android 30:
public interface LocationListener {
void onLocationChanged(@NonNull Location location);
@Deprecated
default void onStatusChanged(String provider, int status, Bundle extras) {}
default void onProviderEnabled(@NonNull String provider) {}
default void onProviderDisabled(@NonNull String provider) {}
}
很明显“默认”二字就是区别。鉴于我和你有同样的问题,我现在需要找到解决方法。或者使用 mohit48 的响应。祝你好运!
重写代码中的方法。 ...
override fun onProviderDisabled(provider: String) {}
override fun onProviderEnabled(provider: String) {}
override fun onStatusChanged(provider: String?, status: Int, extras: Bundle?) {}
...
只需添加实现LocationListener
@Override
public void onProviderEnabled(@NonNull String provider) {
}
@Override
public void onProviderDisabled(@NonNull String provider) {
}
@Override
public void onStatusChanged(String provider, int status, Bundle extras) {
}