代码地址如下:
http://www.demodashi.com/demo/13259.html

前言

讲讲android对于蓝牙耳机连接技术的实现

今天涉及的内容有:

  1. 流程讲解
  2. 新建广播BluetoothReceiver,用于监听处理蓝牙连接过程中各状态
  3. 在MainActivity中调用
  4. 注意的点
  5. 项目结构图和效果图

下面做以讲解

一. 流程讲解

在实现蓝牙耳机链接的时候,需要做一些前期工作,第一步,判断设备是否支持蓝牙。

1.1 设备是否支持蓝牙
   /**设备是否支持蓝牙**/
public boolean isSupportBluetooth() {
mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
if (mBluetoothAdapter != null) {
return true;
}
return false;
}

若支持蓝牙,则需要判断设备蓝牙是否打开

1.2 设备蓝牙是否开启
    /**蓝牙是否已经启动**/
public boolean isBluetoothOpen() {
if (mBluetoothAdapter != null) {
return mBluetoothAdapter.isEnabled();
}
return false;
}

如果蓝牙没有开启,则需要请求开启蓝牙

1.3 请求开启蓝牙
   /**请求启动蓝牙**/
public void requestStartBluetooth(int requestCode,Context context) {
Intent enableBtIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
((MainActivity)context).startActivityForResult(enableBtIntent, requestCode);
}

当然,蓝牙还有一个强制开启的方法:

    /**强制打开蓝牙**/
public void openBluetooth(){
if(isSupportBluetooth()){
mBluetoothAdapter.enable();
}
}

蓝牙开启后,接下来是查询已配对过的设备

1.3 获取配对过的设备列表
   /**查询配对设备**/
public List<BluetoothDevice> checkDevices() {
List<BluetoothDevice> devices=new ArrayList<>();
if(mBluetoothAdapter!=null){
Set<BluetoothDevice> pairedDevices = mBluetoothAdapter.getBondedDevices();
if (pairedDevices != null && pairedDevices.size() > 0) {
for (BluetoothDevice device : pairedDevices) {
devices.add(device);
}
}
}
return devices;
}

若已配对列表中没有你的蓝牙耳机设备,则需要搜索

1.4 搜索新设备
    /**发现新设备**/
public void findBluetoothDevice() {
//其实是启动了一个异步线程,该方法将立即返回一个布尔值,指示发现是否已成功启动。
// 发现过程通常涉及大约12秒的查询扫描,随后是每个找到的设备的页面扫描以检索其蓝牙名称
if(mBluetoothAdapter!=null && mBluetoothAdapter.isEnabled() && !mBluetoothAdapter.isDiscovering()){
if (mBluetoothAdapter.startDiscovery()) {
LogUtil.i("=======已成功启动寻找新设备的异步线程=======");
} else {
LogUtil.i("=======启动寻找新设备的异步线程失败=======");
}
}
}

然后便是进行蓝牙耳机的配对,连接。

以上基本的蓝牙方法,我已经封装到BluetoothManager类中。

在蓝牙耳机的搜索,配对,连接等过程中,我们需要新建一个广播,对各个过程做一个监听。

二. 新建广播BluetoothReceiver,用于监听处理蓝牙连接过程中各状态

下面给出BluetoothReceiver中主要代码:

    @Override
public void onReceive(Context context, Intent intent){
LogUtil.i("=========蓝牙接收处理广播========"+intent.getAction());
BluetoothDevice device;
switch (intent.getAction()) {
case BluetoothA2dp.ACTION_CONNECTION_STATE_CHANGED:
switch (intent.getIntExtra(BluetoothA2dp.EXTRA_STATE, -1)) {
case BluetoothA2dp.STATE_CONNECTING:
device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
LogUtil.i("device: " + device.getName() +" connecting");
break;
case BluetoothA2dp.STATE_CONNECTED:
device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
LogUtil.i("device: " + device.getName() +" connected");
mOnBluetoothListener.deviceConnected(device);
break;
case BluetoothA2dp.STATE_DISCONNECTING:
device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
LogUtil.i("device: " + device.getName() +" disconnecting");
break;
case BluetoothA2dp.STATE_DISCONNECTED:
device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
LogUtil.i("device: " + device.getName() +" disconnected");
break;
default:
break;
}
break;
case BluetoothA2dp.ACTION_PLAYING_STATE_CHANGED:
int state = intent.getIntExtra(BluetoothA2dp.EXTRA_STATE, -1);
switch (state) {
case BluetoothA2dp.STATE_PLAYING:
LogUtil.i("state: playing.");
break;
case BluetoothA2dp.STATE_NOT_PLAYING:
LogUtil.i("state: not playing");
break;
default:
LogUtil.i("state: unkown");
break;
}
break;
case BluetoothDevice.ACTION_FOUND:
device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
int deviceClassType = device.getBluetoothClass().getDeviceClass();
//找到指定的蓝牙设备
if (deviceClassType == BluetoothClass.Device.AUDIO_VIDEO_WEARABLE_HEADSET
|| deviceClassType == BluetoothClass.Device.AUDIO_VIDEO_HEADPHONES) {
LogUtil.i("Found device:" + device.getName()+" address:"+device.getAddress());
if(StringUtil.isNotEmpty(device.getName())){
//添加到设备列表
mOnBluetoothListener.deviceFound(device);
}
}else{//找到可用蓝牙
if(StringUtil.isNotEmpty(device.getName())){
LogUtil.i("=====Found device====11===" + device.getName()+" address:"+device.getAddress());
//添加到设备列表
mOnBluetoothListener.deviceFound(device);
}
}
break;
case BluetoothDevice.ACTION_BOND_STATE_CHANGED:
int bondState = intent.getIntExtra(BluetoothDevice.EXTRA_BOND_STATE, BluetoothDevice.BOND_NONE);
device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
switch (bondState){
case BluetoothDevice.BOND_BONDED: //配对成功
LogUtil.i("Device:"+device.getName()+" bonded.");
//取消搜索,连接蓝牙设备
mOnBluetoothListener.deviceBonded(device);
break;
case BluetoothDevice.BOND_BONDING:
LogUtil.i("Device:"+device.getName()+" bonding.");
break;
case BluetoothDevice.BOND_NONE:
LogUtil.i("Device:"+device.getName()+" not bonded.");
//不知道是蓝牙耳机的关系还是什么原因,经常配对不成功
//配对不成功的话,重新尝试配对
mOnBluetoothListener.deviceBondNone(device);
break;
default:
break; }
break;
case BluetoothAdapter.ACTION_STATE_CHANGED:
state = intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, -1);
switch (state) {
case BluetoothAdapter.STATE_TURNING_ON:
LogUtil.i("BluetoothAdapter is turning on.");
break;
case BluetoothAdapter.STATE_ON:
LogUtil.i("BluetoothAdapter is on.");
// //蓝牙已打开,开始搜索并连接service
// findBluetoothDevice();
// getBluetoothA2DP(); mOnBluetoothListener.blootoothStateOn();
break;
case BluetoothAdapter.STATE_TURNING_OFF:
LogUtil.i("BluetoothAdapter is turning off.");
break;
case BluetoothAdapter.STATE_OFF:
LogUtil.i("BluetoothAdapter is off.");
break;
}
break;
default:
break;
}
}

三. 在MainActivity中调用

3.1 初始化时注册广播,扫描设备列表

       //注册广播
registerReceiver();
//初始化设备列表
initDeviceList();
//发现新设备
findDevices();

其中registerReceiver方法为:

    /**注册广播**/
private void registerReceiver(){
mBluetoothReceiver=new BluetoothReceiver();
IntentFilter filter = new IntentFilter(BluetoothDevice.ACTION_FOUND);
filter.addAction(BluetoothA2dp.ACTION_CONNECTION_STATE_CHANGED);
filter.addAction(BluetoothA2dp.ACTION_PLAYING_STATE_CHANGED);
filter.addAction(BluetoothDevice.ACTION_FOUND);
filter.addAction(BluetoothDevice.ACTION_BOND_STATE_CHANGED);
filter.addAction(BluetoothAdapter.ACTION_STATE_CHANGED);
mContext.registerReceiver(mBluetoothReceiver, filter);
}

findDevices方法为:

    /**发现新设备**/
private void findDevices(){
if(BluetoothManager.getInstance().isSupportBluetooth()&&BluetoothManager.getInstance().isBluetoothOpen()){
//强制打开蓝牙
BluetoothManager.getInstance().openBluetooth(); List<BluetoothDevice>list=BluetoothManager.getInstance().checkDevices();
LogUtil.i("====list=====list=="+list.size()); Iterator<BluetoothDevice> it = list.iterator();
while (it.hasNext()) {
BluetoothDevice device = it.next();
if (device != null&& StringUtil.isEmpty(device.getName())) {
it.remove();
}
}
mDevices.addAll(list);
myAdapter.notifyDataSetChanged();
}
}

3.2 点击设备列表去连接蓝牙耳机或者开启蓝牙扫描

        myAdapter.setOnRecyclerItemClickListener(new MyAdapter.OnRecyclerItemClickListener() {
@Override
public void onRecyclerClick(View view, int position) {
BluetoothDevice device= mDevices.get(position);
if(!BluetoothManager.getInstance().isSupportBluetooth()){
ToastUtil.showShortToast(mContext,"本设备不支持蓝牙");
return;
}
if(!BluetoothManager.getInstance().isBluetoothOpen()){
//去启动蓝牙
BluetoothManager.getInstance().requestStartBluetooth(REQUEST_ENABLE_BT,mContext);
}else{
LogUtil.i("====开始配对=======");
//绑定BluetoothA2DP,获得service
BluetoothManager.getInstance().getBluetoothA2DP(mContext);
//开始配对
BluetoothManager.getInstance().createBond(device);
}
}
});

3.3 关闭资源

退出app的时候需要关闭蓝牙耳机连接

       //注销蓝牙链接
BluetoothManager.getInstance().disableAdapter();

注销广播

       //注销广播
if(mBluetoothReceiver!=null){
mContext.unregisterReceiver(mBluetoothReceiver);
}

当然,你还可以考虑是否需要关闭蓝牙

//关闭蓝牙
BluetoothManager.getInstance().closeBluetooth();

四. 注意的点

蓝牙耳机的连接需要蓝牙权限,所以你得在你的mainfast.xml中以下权限设置:

    <uses-permission android:name="android.permission.BLUETOOTH" />
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />

五. 项目结构图和效果图

项目结构图

运行效果图



Android实现蓝牙耳机连接

代码地址如下:
http://www.demodashi.com/demo/13259.html

注:本文著作权归作者,由demo大师代发,拒绝转载,转载需要作者授权

Android实现蓝牙耳机连接的更多相关文章

  1. Android跟蓝牙耳机建立连接有两种方式

    Android 跟蓝牙耳机建立连接有两种方式. 1. Android 主动跟蓝牙耳机连BluetoothSettings 中和蓝牙耳机配对上之后, BluetoothHeadsetService 会收 ...

  2. Android实现主动连接蓝牙耳机

    在Android程序中可以实现自动扫描蓝牙.配对蓝牙.建立数据通道. 蓝牙分不同类型,可以参考(http://gqdy365.iteye.com/admin/blogs/2229304) 可以入下面方 ...

  3. Android蓝牙A2DP连接实现

    代码地址如下:http://www.demodashi.com/demo/14624.html 开发环境: 开发工具:Androidstudio 适配机型:honor8(Android6.0), 坚果 ...

  4. 71.Android之长连接实现

    转载:http://blog.csdn.net/qq_23547831/article/details/51690047 本文中我们将讲解一下App的长连接实现.一般而言长连接已经是App的标配了,推 ...

  5. Android平台上长连接的实现

    Android 平台上长连接的实现 为了不让 NAT 表失效,我们需要定时的发心跳,以刷新 NAT 表项,避免被淘汰. Android 上定时运行任务常用的方法有2种,一种方法用 Timer,另一种是 ...

  6. [转]Mac OS X 下部分Android手机无法连接adb问题之解决方案

    时至当今,Android山寨手机厂商已如此之多,能修改和个性化定制Android OS的能人已是多如牛毛,有的牛人修改Android系统只会影响所修改的点,不会影响其它,然后还有的就不多说了,总之做的 ...

  7. 小米2及其他Android手机无法连接mac解决方案

    一般的android连接mac 很方便不用安装驱动就可以啦,可是不知道为什么二般情况下有的android手机(小米2,华为等)就是连接不上,下来就说说二般情况下如何连接. 1.关于本机-->更多 ...

  8. Android检测网络连接

    Android检测网络连接 import android.app.AlertDialog; import android.content.Context; import android.content ...

  9. Mac OS X 下部分Android手机无法连接adb问题之解决方案

    [原文]  时至当今,Android山寨手机厂商已如此之多,能修改和个性化定制Android OS的能人已是多如牛毛,有的牛人修改Android系统只会影响所修改的点,不会影响其它,然后还有的就不多说 ...

随机推荐

  1. foreach 与 Linq的 Select 效率问题

    Resharper 是一个非常强大的C#编程辅助工具,有着非常强的提示功能,代码纠正,代码简化等等 在编码过程中注意到这么一件事,可能是大家经常会遇到的: 遍历某个集合,然后经过处理生成另外一个集合, ...

  2. C指针详解

    前言:复杂类型说明 要了解指针,多多少少会出现一些比较复杂的类型,所以我先介绍一下如何完全理解一个复杂类型,要理解复杂类型其实很简单,一个类型里会出现很多运算符,他们也像普通的表达式一样,有优先级,其 ...

  3. sqlmap的小试牛刀

    这次算是一次用sqlmap的例子,写的很水. 目的:通过工具扫描到了后台的数据库的地址(如下图),想通过sqlmap找到数据库的用户和密码进入到数据库(首先可以尝试一下root:root一般都是这个情 ...

  4. E - A strange lift 【数值型BFS+上下方向】

    There is a strange lift.The lift can stop can at every floor as you want, and there is a number Ki(0 ...

  5. ASP.NET Core 2.2 基础知识(十一) ASP.NET Core 模块

    ASP.NET Core 应用与进程内的 HTTP 服务器实现一起运行.该服务器实现侦听 HTTP 请求,并在一系列请求功能被写到 HttpContext 时,将这些请求展现到应用中. ASP.NET ...

  6. leetcode171 Excel Sheet Column Number

    题意: A -> 1 B -> 2 C -> 3 ... Z -> 26 AA -> 27 AB -> 28 思路:找规律,挺简单的···可是小地方错了一些搞了半天 ...

  7. 【分块】bzoj3226 [Sdoi2008]校门外的区间

    题解见 : http://blog.csdn.net/iamzky/article/details/41088151 ORZ ZKY 2个懒标记:是否翻转,覆盖成了什么. 怎么处理一个块上有两个标记的 ...

  8. Spark1.4安装问题

    1)按照<大数据Spark企业级实战>第2章中的方法构建Spark集群,最后发现master可以正常启动,但是worker却都没有启动,原因是不能直接使用在slave模版文件 slaves ...

  9. IOS,苹果内购和添加广告

    内购——应用内购买 通过苹果应用程序商店有三种主要赚钱的方式: 直接收费(与国内大部分用户的消费习惯相悖) 广告(降低用户体验 应用程序名称带Lite可以添加广告) O2O -> Online推 ...

  10. 乐观锁-version的使用

    出处:http://chenzhou123520.iteye.com/blog/1863407 乐观锁介绍: 乐观锁( Optimistic Locking ) 相对悲观锁而言,乐观锁假设认为数据一般 ...