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

时间:2014-11-06 22:27:41

标签: android bluetooth-lowenergy noclassdeffounderror android-bluetooth buildpath

我向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代码...... < / p>

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" />    

第一个问题,我上面的假设是能够在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 Order and Export

Eclipse Libraries

4 个答案:

答案 0 :(得分:2)

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

答案 1 :(得分: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) {

            }
        };

    }
}

答案 2 :(得分:0)

在&lt; 4.3设备上获取此异常是正常的,因为BLE不存在,因此您的编译代码无法在OS中找到相应的类。 你的构建路径没有错。

最佳解决方案确实是在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" />

答案 3 :(得分:0)

现在您使用的是旧版Android可能无法使用的API,请确保使用适当的支持库打包该应用。

例如,请参阅此android sample BluetoothLeGattSample。它使用以下支持lib。

    dependencies {
       // Add the support lib that is appropriate for SDK 18
       compile "com.android.support:support-v13:19.0.+"
    }

see android docs for eclipse instructions for support library setup