Android LocationListener,onStatusChanged 和 onProviderDisabled 上的 AbstractMethodError

问题描述 投票:0回答:6

我有一个带有按钮的简单活动,它使用 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 打开并按下按钮时它不会崩溃。

从文档来看,这些方法被标记为默认,所以我不需要实现它们。

这种行为有什么原因吗?

android locationmanager
6个回答
48
投票

只需在代码末尾添加这三个函数即可:

@Override
public void onProviderEnabled(@NonNull String provider) {

}

@Override
public void onProviderDisabled(@NonNull String provider) {

}

@Override
public void onStatusChanged(String provider, int status, Bundle extras) {
    
}

并编写您想要在状态更改时执行的代码


24
投票

我对上面的答案并不完全满意,决定详细说明一下。

正如 @sunlover3 正确指出的那样,有问题的方法的实现已更改。在 API 30 之前,它们是抽象的,这意味着如果compileSdk 不是 30,OP 的代码将给出编译错误(这些方法必须已实现)。但在API 30中,他们添加了空的默认方法的实现,这意味着如果compileSdk是30,编译器检查这些方法并且没有给出错误,则它们已经实现了。

但是,如果在版本低于 30 的旧操作系统上运行带有compileSdk 30 的应用程序,这些方法的实现仍然是旧的,i.o.没有,它们是抽象的!因此,出现错误。

老实说,这是一个明显的 SDK 开发错误,这样的事情绝对不能发生。最坏的情况应该是出现编译时警告,表明仍应重写这些方法以实现旧版本的兼容性。我希望在这个话题上比我不那么懒的人能够向 google tracker 报告这个问题:)


7
投票

这是我的情况。在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 的响应。祝你好运!


3
投票

重写代码中的方法。 ...

override fun onProviderDisabled(provider: String) {}
override fun onProviderEnabled(provider: String) {}
override fun onStatusChanged(provider: String?, status: Int, extras: Bundle?) {}
...

3
投票

我遇到了同样的问题,我设法通过覆盖上面建议的 onStatusChanged() 来解决它。

但是,Android 团队可能正在研究一些东西,据我测试,我发现有 this 接口可以解决问题而无需覆盖。

您可以在此处查看androidx.core.location包的摘要。

为了在您的项目中使用这些新类,您应该使用 androidx.core 的最新 beta 版本。请阅读此处


1
投票

只需添加实现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) {
    
}
© www.soinside.com 2019 - 2024. All rights reserved.