Android12中蓝牙权限适配

文章详细介绍了在Android12中针对蓝牙权限的更新,包括AndroidManifest.xml中的配置变化,如新增的BLUETOOTH_SCAN、BLUETOOTH_ADVERTISE、BLUETOOTH_CONNECT权限。同时,文中提供了蓝牙状态监听的BroadcastReceiver实现,并展示了如何在代码中动态申请Android12的蓝牙权限。强调了在Android12中,不申请定位权限时需设置usesPermissionFlags为neverForLocation,否则可能无法搜索到设备。最后指出,无论在哪个Android版本,未开启蓝牙或缺少权限都将无法正常使用蓝牙功能。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

AndroidManifest.xml

<!--maxSdkVersion=30适配到Android11 -->
<uses-permission
    android:name="android.permission.BLUETOOTH"
    android:maxSdkVersion="30" />
<uses-permission
    android:name="android.permission.BLUETOOTH_ADMIN"
    android:maxSdkVersion="30" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<!--Android12新增蓝牙权限 -->
<!-- Android 12在不申请定位权限时,必须加上android:usesPermissionFlags="neverForLocation",否则搜不到设备 -->
<uses-permission android:name="android.permission.BLUETOOTH_SCAN"
        android:usesPermissionFlags="neverForLocation"
        tools:targetApi="s" />
<uses-permission android:name="android.permission.BLUETOOTH_ADVERTISE" />
<uses-permission android:name="android.permission.BLUETOOTH_CONNECT" />

代码

//获取蓝牙适配器
public static BluetoothAdapter mBluetoothAdapter = null;
    //蓝牙状态监听广播
    public class BlueToothReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            switch (intent.getAction()) {
                case BluetoothAdapter.ACTION_STATE_CHANGED:
                    int blueState = intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, 0);
                    switch (blueState) {
                        case BluetoothAdapter.STATE_TURNING_ON:
                            Log.e("onReceive", "---------蓝牙正在打开中");
                            break;
                        case BluetoothAdapter.STATE_ON:
                            Log.e("onReceive", "---------蓝牙已经打开");
                            mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
                            break;
                        case BluetoothAdapter.STATE_TURNING_OFF:
                            Log.e("onReceive", "---------蓝牙正在关闭中");
                            break;
                        case BluetoothAdapter.STATE_OFF:
                            Log.e("onReceive", "---------蓝牙已经关闭");
                            break;
                    }
                    break;
            }
        }
    }
    private IntentFilter makeFilter() {
        IntentFilter filter = new IntentFilter();
        filter.addAction(BluetoothAdapter.ACTION_STATE_CHANGED);
        return filter;
    }
    //Android12蓝牙权限申请
    private boolean bluePermission(){
    //compileSdkVersion项目中编译SDK版本大于30申请以下权限可使用
    //Manifest.permission.BLUETOOTH_SCAN、Manifest.permission.BLUETOOTH_ADVERTISE、Manifest.permission.BLUETOOTH_CONNECT
    //若小于30可以直接使用权限对应的字符串
        if (Build.VERSION.SDK_INT>30){
            if (ContextCompat.checkSelfPermission(this,
                    "android.permission.BLUETOOTH_SCAN")
                    != PERMISSION_GRANTED
                    || ContextCompat.checkSelfPermission(this,
                    "android.permission.BLUETOOTH_ADVERTISE")
                    != PERMISSION_GRANTED
                    || ContextCompat.checkSelfPermission(this,
                    "android.permission.BLUETOOTH_CONNECT")
                    != PERMISSION_GRANTED){
                ActivityCompat.requestPermissions(this,new String[]{
                                "android.permission.BLUETOOTH_SCAN",
                                "android.permission.BLUETOOTH_ADVERTISE",
                                "android.permission.BLUETOOTH_CONNECT"}, 1);
                return false;
            }
        }
        return true;
    }
    private void openBlueTooth(){
        if (bluePermission())
            mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
        else
            return;
        BlueToothReceiver btlr = new BlueToothReceiver();
        this.registerReceiver(btlr, makeFilter());
        if (!mBluetoothAdapter.isEnabled()) {// 判断是否打开蓝牙
            showHandlerToast("请先开启蓝牙!");
            //弹出对话框提示用户是后打开
            Intent intent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
            //startActivityForResult(intent, SEARCH_CODE);
            // 不做提示,强行打开
            mBluetoothAdapter.enable();
        }
    }

说明:

必须按照这两步申请蓝牙权限,首先在androidManifest.xml中配置蓝牙权限,Android11及以下android.permission.BLUETOOTH、android.permission.BLUETOOTH_ADMIN申请这两个,Android12中蓝牙权限进行新增android.permission.BLUETOOTH_SCAN、android.permission.BLUETOOTH_ADVERTISE、android.permission.BLUETOOTH_CONNECT使用这三个;其次在代码中需要动态申请Android12的蓝牙权限,测试发现Android12中动态申请权限后则蓝牙直接开启,通过对蓝牙状态的监听获取蓝牙适配器:mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();

注意:不管是Android12还是Android12以下不打开蓝牙都是无法获取蓝牙适配器:mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();打开蓝牙则可以获取到蓝牙适配器,但若没有申请蓝牙所需权限,蓝牙功能也是无法使用!!!二者缺一不可

### 关于 Android 14 上蓝牙功能的适配 #### API 变化 对于 `BluetoothDevice` 类,在 Android 14 中新增了一个名为 `getPackageNameOfBondingApplication()` 的方法,该方法用于获取绑定应用程序的包名[^1]。 #### 权限设置 自 Android 12 开始引入的新权限机制继续影响着更高版本的操作系统。当应用尝试访问蓝牙特性时,必须声明并请求必要的权限。特别是针对蓝牙扫描操作,需确保已获得位置权限以及蓝牙特定权限。如果第三方 APP 出现启动即闪退的情况,则可能是由于未能正确处理这些新的权限需求所引起的[^3]。 为了兼容 Android 14 并实现良好的用户体验,开发者应当仔细审查官方文档中的权限更改部分,并按照指导更新应用内的权限管理逻辑[^4]。 #### 常见问题解决 - **设备连接数量受限**:随着操作系统版本升级至 Android 7.0及以上,支持更高级别的 BLE 协议栈(如蓝牙4.2 和 5.0),这不仅提高了数据传输效率还增加了稳定的同时多设备连接能力[^2]。 - **应用崩溃或异常行为**:若遇到此类情况,建议先确认是否遵循了最新的 SDK 版本要求(`targetSdkVersion`),并且适当调整代码以适应任何可能存在的平台级改动;另外还需关注是否有其他依赖库也需要同步更新来保持一致性。 ```java // 请求必要权限示例 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) { requestPermissions(new String[]{ Manifest.permission.BLUETOOTH_SCAN, Manifest.permission.BLUETOOTH_CONNECT, Manifest.permission.ACCESS_FINE_LOCATION}, REQUEST_CODE); } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值