我有一个问题,它涉及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
答案 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版本中不存在的类的不必要错误。
希望能帮到你。