旧版Android SDK上的java.lang.NoClassDefFoundError问题

9
我将应用程序的一个版本发布到了Google Play上,今天早上发现有不少用户对此不满。该应用程序的最新版本支持蓝牙低功耗(BTLE)心率监测器。该应用程序在Android 4.3和4.4上运行良好,但在4.0、4.1和4.2上崩溃,并出现以下错误。
FATAL EXCEPTION: main
java.lang.NoClassDefFoundError: com.eiref.boatcoach.MainActivity
    at com.eiref.boatcoach.WhatToDo.onClick(WhatToDo.java:274)
    at android.view.View.performClick(View.java:4204)
    at android.view.View$PerformClick.run(View.java:17355)
    at android.os.Handler.handleCallback(Handler.java:725)
    at android.os.Handler.dispatchMessage(Handler.java:92)
    at android.os.Looper.loop(Looper.java:152)
    at android.app.ActivityThread.main(ActivityThread.java:5132)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:511)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:793)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:560)
    at dalvik.system.NativeStart.main(Native Method)

在类似以下的简单Onclick中创建Intent时会出现错误...
public void onClick(View v) {
        Intent i = new Intent(this, MainActivity.class);
        startActivity(i);
}

在匆忙购买了一台4.2平板电脑以便能够复制该问题后,我得出结论:这与新版本的应用程序支持蓝牙LE有关,该功能在SDK 4.3及更高版本中启用。如果我删除MainActivity中所有关于蓝牙的引用,则在4.2及更早版本的设备上崩溃问题将消失。
根据文档的理解,我们可以编写一个包含蓝牙LE功能的应用程序,并在旧设备上运行,只要小心不要使用以下代码执行BTLE代码...
if (!getPackageManager().hasSystemFeature(PackageManager.FEATURE_BLUETOOTH_LE)) return;
    BluetoothManager manager = (BluetoothManager) getSystemService(BLUETOOTH_SERVICE);
    mBluetoothAdapter = manager.getAdapter();
//etc.

因此,我的manifest.xml文件不会包括以下内容,因为这会阻止较老设备的下载,而我显然希望尽可能地维护单一代码库...
<uses-feature android:name="android.hardware.bluetooth_le" android:required="true" />    

第一个问题,我上面的假设能够在SDK 4.3之前的SDK中包含BTLE代码是正确的吗?如果不是,我真的需要创建两个版本的应用程序......一个供使用4.3及更高版本的人使用,另一个供其他所有人使用吗?

关于java.lang.NoClassDefFoundError,有很多StackOverflow帖子,我认为我已经阅读了大部分相关帖子。许多建议我检查Java Build Path,以确保已选中Android Private Libraries和Android Dependencies。它们已经被选中了。一些人建议将gen文件夹移到src文件夹之前,但这似乎没有什么区别。

我想发布Eclipse Java Build Path的图像,但由于这是我的第一篇文章,我没有10个声望点来插入图像,因此这里是我遵循的另一篇文章... Android java.lang.NoClassDefFoundError

所以,第二个问题,有什么其他想法可能导致构建路径出错吗?

非常感谢您提前的帮助。


更新...不知怎么地,通过提问我得到了足够的积分来发布Java构建路径的图像。正如@Ashoke指出的那样,我认为这与错误的构建路径或支持库有关。

Eclipse Order and Export

Eclipse Libraries


你的 onClick 内部代码在 >= 4.3 上是否运行相同,还是根据 Android 版本有另一条代码路径? - Michael Krause
你绝对不需要两个版本的应用程序。至于uses-feature标签,我建议您*包括它,但设置android:required="false"。我并不认为这是根本问题,但无论如何这样做都是好的。 - Kevin Coppock
1
你的主要活动是否直接实现了任何BLE回调方法?或者你是否有任何静态引用它们的地方?我的所有BLE相关内容都在一个扩展类中,继承自我的标准内容。如果操作系统版本太低,我根本不会实例化这个类。 - Ifor
@ifor - 谢谢!将所有的BLE代码放在一个单独的类中,然后只有在支持BLE时实例化该类似乎解决了问题。 - Dan Eiref
MainActivity 的清单条目是什么? - Code-Apprentice
显示剩余3条评论
4个回答

2

尝试将所有与Ble相关的代码放入一个单独的类中,仅在具有必要API级别的设备上进行实例化。我认为如果没有这个,回调可能会导致您的问题。


1
我发现在活动中静态定义 le 回调会出错,但是将其包含在一个带有 API 检查的函数中则不会产生相同的错误。
而不是:
    @TargetApi(Build.VERSION_CODES.LOLLIPOP){
    public class RouteMapActivity extends ActionBarActivity

    private BluetoothAdapter.LeScanCallback mScanCallback = new ScanCallback()    {...}

我使用了:

private void setUpLeCallbacks(){


    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        settings = new ScanSettings.Builder()
                .setScanMode(ScanSettings.SCAN_MODE_LOW_LATENCY)
                .build();

        for (BTDeviceName device : mTestPointsToRead) {
            ScanFilter filter = new ScanFilter.Builder().setDeviceAddress(device.le_serial).build();
            filters.add(filter);
        }

        mScanCallback = new ScanCallback() {
            @Override
            public void onScanResult(int callbackType, ScanResult result) {
                super.onScanResult(callbackType, result);
            }

            @Override
            public void onScanFailed(int errorCode) {
                super.onScanFailed(errorCode);
            }
        };
    }else {
        mLeScanCallback= new BluetoothAdapter.LeScanCallback() {

            @Override
            public void onLeScan(final BluetoothDevice device, int rssi, byte[] scanRecord) {

            }
        };

    }
}

你是如何添加全局变量而不受dalivk的VFY警告的影响的? - Mark Gilchrist
1
@TargetApi(Build.VERSION_CODES.LOLLIPOP) 覆盖了 lint 错误。 - TacoEater

0

在<4.3设备上出现此异常是正常的,因为BLE不存在,所以您编译的代码无法在操作系统中找到相应的类。您的构建路径没有问题。

最好的解决方案确实是在运行时使用if语句保护您的BLE代码,测试BLE功能。您还应该在if语句中过滤操作系统版本,例如:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2) {...}

你应该为这个功能声明 android:required="false",这意味着如果设备上存在该功能,应用程序会优先使用它,但是如果必要的话,应用程序也可以在没有指定功能的情况下正常运行。
<uses-feature android:name="android.hardware.bluetooth_le" android:required="false" />

感谢您的反馈。然而,问题在于MainActivity.class(带有我的蓝牙代码)在Android版本<4.3中找不到(导致异常),因此我们甚至无法进入“if”子句。 - Dan Eiref
我认为你还应该添加一个操作系统检查,以确保在编译时不会针对低版本的操作系统到达此代码(我已更新答案)。 - Murphy

0

Ashoke - 感谢您的建议。我尝试将android-support-v13.jar添加到Eclipse构建路径中,但没有成功。 - Dan Eiref

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接