没有初始化的字段声明和Java中的用法

时间:2016-06-28 16:42:28

标签: java android android-bluetooth android-api-levels

我有一个问题,它涉及java编译器行为以及android的不同API。

我有代表蓝牙LE连接的课程。我在这个类中检查API的版本,以了解扫描我必须使用的设备的方法。这是:

public class BleConnector{

    private MyScanCallback _scanCallback;


private LeScanHandle _leScanHndl;
private BluetoothAdapter _bluetoothAdapter;

/**
 * Initializes new instance of BLE connector
 */
public BleConnector()
{
    //Creating handle for device scan callback;
    _leScanHndl = new LeScanHandle();

    //Check version of android api
    //If it is bigger of equal to 21
    //create scan callback for higher versions
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP)
    {
        _scanCallback = new ScanCallback(_leScanHndl);
    }
}


public void startScan(Context context) throws PermissoinException
{
    final BluetoothManager bluetoothManager = (BluetoothManager) context.getSystemService(Context.BLUETOOTH_SERVICE);
    _bluetoothAdapter = bluetoothManager.getAdapter();

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP)
    {
        if (ContextCompat.checkSelfPermission(context, Manifest.permission.ACCESS_COARSE_LOCATION) == PackageManager.PERMISSION_GRANTED)
        {
            _bluetoothAdapter.getBluetoothLeScanner().startScan(_scanCallback);
        }
        else
        {
             throw new PermissoinException("Location permission does not allowed");
        }
    }
    else
    {
        _bluetoothAdapter.startLeScan(_leScanHndl);
    }
    //TODO notify
}

public void stopScan()
{
    if (_bluetoothAdapter != null)
    {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP)
        {
           _bluetoothAdapter.getBluetoothLeScanner().stopScan(_scanCallback);
        } else
        {
            _bluetoothAdapter.stopLeScan(_leScanHndl);
        }
    }

    //TODO notify
}

private void connectToDevice(BluetoothDevice device)
{
    //TODO notify

   //TODO connect gatt
}

Class MyScanCallback标有@TargetAPI 21,当然,如果我的设备版本低于21,我不想使用它

正如您所看到的, _scanCallback 的所有用法和初始化都在 if 语句中。所以我的问题是:

如果没有初始化和 _scanCallback 的用法,我会在api <21的设备上使用声明private MyScanCallback _scanCallback;遇到任何麻烦吗?

谢谢!

UDT

哦,我没有说MyScanCallback继承了来自21 API级别public class MyScanCallback extends ScanCallback

的ScanCallback

1 个答案:

答案 0 :(得分:2)

@TargetApi 21只是为了防止使用任何新API的编译错误。实际上要避免代码被执行,它必须像在几个地方一样完成构建版本检查。

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

因此,在这种情况下,肯定会创建一个MyScanCallback变量,但它不会在api <21的设备上实例化,因为你已将它包装在版本检查构造函数中。

正如您所提到的那样,它正在使用API​​ 21中引入的ScanCallback,那么在旧版本中使用它肯定会出现问题。

为了摆脱这种情况,我建议创建一个抽象类,如下所示:

public abstract class BleConnector{

public abstract void startScan(Context context);

}

然后创建两个类

public class BleConnectOldApi extends BleConnector{
public void startScan(Context context) throws PermissoinException {
//Your code here
}
}

@TargetApi(21)
public class BleConnect21Api extends BleConnector{

private MyScanCallback _scanCallback;

public void startScan(Context context) throws PermissoinException {
//Your code here
}
}

最后有一个工厂类

public class BleConnectorFactory{

    public static BleConnector getBleConnector() {

        if (android.os.Build.VERSION.SDK_INT >= 21) {
            return new BleConnect21Api();
        }else{
            return new BleConnectOldApi();
        }

    }
}

通过这种方式,您可以防止使用旧API版本中不存在的类的不必要错误。

希望能帮到你。