3

I am very new to android dev. I've watched a few youtube video's and read some documentation. Today is like day 5, so be patient with me. I am trying to build a BluetoothLE app. I stole some code from here, but when I try to run in on my device. I get a "Unfortunately, (app name) has stopped" message. After commenting out different blocks of code I have pinned it down to the declaration of:

private ScanCallback mScanCallback = new ScanCallback() {};

If I include that line, it crashes. If I don't, then it lives. Here is the full Java Class:

package com.tremor.creech.bluetoothExample;

import android.annotation.TargetApi;
import android.bluetooth.BluetoothAdapter;
import android.bluetooth.BluetoothDevice;
import android.bluetooth.BluetoothGatt;
import android.bluetooth.le.BluetoothLeScanner;
import android.bluetooth.le.ScanCallback;
import android.bluetooth.le.ScanFilter;
import android.bluetooth.le.ScanSettings;
import android.os.Bundle;
import android.os.Handler;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;

import java.util.List;
@TargetApi(21)
public class MainActivity extends AppCompatActivity {

    private BluetoothAdapter mBluetoothAdapter;
    private int REQUEST_ENABLE_BT = 1;
    private Handler mHandler;
    private static final long SCAN_PERIOD = 10000;
    private BluetoothLeScanner mLEScanner;
    private ScanSettings settings;
    private List<ScanFilter> filters;
    private BluetoothGatt mGatt;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    private ScanCallback mScanCallback = new ScanCallback() { }; //If I include this line, then it crashes. If I take it out, then I see my Hello World activity.
}

LogCat errors:

09-15 19:16:17.503    1878-1878/com.roberts.croberts.mantis E/dalvikvm﹕ Could not find class 'com.roberts.croberts.mantis.MainActivity$1', referenced from method com.roberts.croberts.mantis.MainActivity.<init>
09-15 19:16:17.523    1878-1878/com.roberts.croberts.mantis E/AndroidRuntime﹕ FATAL EXCEPTION: main
    Process: com.roberts.croberts.mantis, PID: 1878
    java.lang.NoClassDefFoundError: com.roberts.croberts.mantis.MainActivity$1
            at com.roberts.croberts.mantis.MainActivity.<init>(MainActivity.java:154)
            at java.lang.Class.newInstanceImpl(Native Method)
            at java.lang.Class.newInstance(Class.java:1208)
            at android.app.Instrumentation.newActivity(Instrumentation.java:1061)
            at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2119)
            at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2252)
            at android.app.ActivityThread.access$800(ActivityThread.java:139)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1200)
            at android.os.Handler.dispatchMessage(Handler.java:102)
            at android.os.Looper.loop(Looper.java:136)
            at android.app.ActivityThread.main(ActivityThread.java:5103)
            at java.lang.reflect.Method.invokeNative(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:515)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:790)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:606)
            at dalvik.system.NativeStart.main(Native Method)
09-15 19:16:18.023      461-717/? E/Diag_Lib﹕ Thread state: conn_id=0, state=RX_THREAD_WAIT_READ
09-15 19:16:18.023      461-717/? E/Diag_Lib﹕ Thread state: conn_id=0, state=RX_THREAD_CLIENT_TX
09-15 19:16:18.023      461-717/? E/Diag_Lib﹕ qmi_qmux: TX/RX - RX 30 bytes on conn_id=0
09-15 19:16:18.023      461-717/? E/Diag_Lib﹕ 01 1D 00 80 03 01 04 00 00 51 00 11 00 10 03 00
09-15 19:16:18.023      461-717/? E/Diag_Lib﹕ BA 0E 00 11 08 00 BE 08 00 07 00 00 00 00
09-15 19:16:18.023      461-717/? E/Diag_Lib﹕ qmuxd: TX message on fd=29, to qmux_client_id=2, len=64
09-15 19:16:18.023      461-717/? E/Diag_Lib﹕ Thread state: conn_id=0, state=RX_THREAD_WAIT_POLL
09-15 19:16:18.033      454-454/? E/Parcel﹕ Reading a NULL string not supported here.
09-15 19:16:18.033      454-454/? E/Parcel﹕ Reading a NULL string not supported here.
09-15 19:16:18.033      454-454/? E/Parcel﹕ Reading a NULL string not supported here.
09-15 19:16:18.033      454-454/? E/Parcel﹕ Reading a NULL string not supported here.
4

3 回答 3

9

不幸的是,要在 android 下构建 BLE 应用程序,您必须将 android Jelly Bean MR2(18) + Kitkat(19) 和 Lollipop (21+) 分开。

因为你才刚刚开始,所以我给你概述一下如何检查。

首先为JellyBean/Kitkatlollipop声明这些变量。

//For Jelly Bean MR2 and Kitkat
private BluetoothAdapter.LeScanCallback leScanCallback;

//For Lollipop.
private BluetoothLeScanner bluetoothLeScanner;
private ScanCallback scanCallback;

由于它们没有启动,因此运行应用程序时不会出现错误。为了启动目标 api 是什么,您可以使用Build

if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
    //JellyBean/Kitkat
} else {
    //lollipop+
}

另外不要忘记检查手机是否真的具有 Ble 功能并且是否需要将 uses 功能添加到清单中:

<uses-feature android:name="android.hardware.bluetooth_le" android:required="true"/>

并从PackageManager检查:

boolean hasBle = getPackageManager().hasSystemFeature(PackageManager.FEATURE_BLUETOOTH_LE);

必须执行此检查,因为某些手机(如 Galaxy nexus)可用于 jelly bean MR2 但不具有 BLE 功能)

最后的建议是,要非常小心回调泄漏并正确停止/删除回调;)

于 2015-09-16T16:22:45.087 回答
2

在此线程上回复为时已晚,但我想分享我的想法,以克服 API 21 以下设备上的崩溃。ScanCallback该类具有 API 21 的支持。因此,当您编写代码以在扫描代码中实现 ScanCallback 时,它将导致情人 API 崩溃。我已经通过以下方式修复了它:

我创建了一个新的抽象类,它扩展了 ScanCallback 类,如下所示:

@TargetApi(Build.VERSION_CODES.LOLLIPOP)  
public abstract class AppScanCallback extends ScanCallback {}

现在我在我的 BluetoothService 类中使用这个类实例,如下所示:

@RequiresApi(Build.VERSION_CODES.LOLLIPOP)
private AppScanCallback mScanCallback;

并使用此变量如下:

public void startBleScan() {
    if (isEnabled()) {
        if (Build.VERSION.SDK_INT < 21) {
            _bluetoothAdapter.startLeScan(mLeScanCallback);
        } else {
            mLEScanner = _bluetoothAdapter.getBluetoothLeScanner();
            settings = new ScanSettings.Builder()
                    .build();
            filters = new ArrayList<>();

            mScanCallback = new AppScanCallback() {
                @Override
                public void onScanResult(int callbackType, ScanResult result) {
                    if (Build.VERSION.SDK_INT >= 21) {
                        BluetoothDevice btDevice = result.getDevice();
                        onLeScanResult(btDevice, result.getScanRecord().getBytes());
                    }
                }

                @Override
                public void onBatchScanResults(List<ScanResult> results) {
                    for (ScanResult sr : results) {
                        Log.i("ScanResult - Results", sr.toString());
                    }
                }
                @Override
                public void onScanFailed(int errorCode) {
                    Log.e("Scan Failed", "Error Code: " + errorCode);
                }
            };
            mLEScanner.startScan(filters, settings, mScanCallback);
        }

    }
}

并停止扫描

public void stopBLEScan() {;
        if (Build.VERSION.SDK_INT < 21) {
            _bluetoothAdapter.stopLeScan(mLeScanCallback);
        } else {
            mLEScanner.stopScan(mScanCallback);
        }
}

希望它会帮助你。

于 2018-12-17T06:38:03.707 回答
1

而不是这个

   private ScanCallback mScanCallback = new ScanCallback() { }; 

尝试这个

  private BluetoothAdapter.LeScanCallback mLeScanCallback = new BluetoothAdapter.LeScanCallback() {};
于 2015-09-16T05:54:28.147 回答