较旧的Android SDK版本上的java.lang.NoClassDefFoundError

Dan*_*ref 8 android noclassdeffounderror buildpath bluetooth-lowenergy android-bluetooth

我向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)
Run Code Online (Sandbox Code Playgroud)

在类似于以下内容的简单Onclick中创建Intent时发生错误...

public void onClick(View v) {
        Intent i = new Intent(this, MainActivity.class);
        startActivity(i);
}
Run Code Online (Sandbox Code Playgroud)

在冲出并购买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.
Run Code Online (Sandbox Code Playgroud)

因此,我的manifest.xml并没有包括以下内容,因为它会阻止下载到旧设备,我当然希望如果可能的话,保持一个代码库...

<uses-feature android:name="android.hardware.bluetooth_le" android:required="true" />    
Run Code Online (Sandbox Code Playgroud)

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

有很多关于java.lang.NoClassDefFoundError的StackOverflow帖子,我想我已经阅读了大部分相关内容.许多人建议我检查Java Build Path以确保检查Android Private Libraries和Android Dependencies.他们是.有人建议在src文件夹之前移动gen文件夹,但这似乎没有什么区别.

我发布了Eclipse Java Build Path的图片,但由于这是我的第一篇文章,我没有插入图片所需的10个声望点,所以这是我跟随的另一篇文章... Android java.lang.NoClassDefFoundError

那么,第二个问题,关于构建路径可能出现什么问题的任何其他想法?

提前谢谢了.


更新...以某种方式通过询问一个问题我得到了足够的点来发布java构建路径的图像.正如@Ashoke指出的那样,我确实认为它与错误的构建路径或支持库有关.

Eclipse订单和导出

Eclipse库

Mur*_*phy 0

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

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

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2) {...}
Run Code Online (Sandbox Code Playgroud)

您应该声明android:required="false"此功能,这意味着应用程序更愿意使用设备上存在的功能,但如果需要,它被设计为在没有指定功能的情况下运行。

<uses-feature android:name="android.hardware.bluetooth_le" android:required="false" />
Run Code Online (Sandbox Code Playgroud)