本文转载自:http://blog.sina.com.cn/s/blog_631e3f2601012ixi.html

Android蓝牙串口通讯

闲着无聊玩起了Android蓝牙模块与单片机蓝牙模块的通信,简单思路就是要手机通过蓝牙发送控制指令给单片机,并作简单的控制应用。单片机的蓝牙模块连接与程序暂且略过,此文主要描述Android手机蓝牙客户端遇到的那点破事。进入正题:

连接蓝牙设备——蓝牙客户端:

Android手机一般以客户端的角色主动连接SPP协议设备(接上蓝牙模块的数字传感器),客户端连接流程是:

1.使用registerReceiver注册BroadcastReceiver来获取蓝牙状态、搜索设备等消息;

    private BroadcastReceiver searchDevices = new BroadcastReceiver() {

public void onReceive(Context context, Intent intent) {

String action = intent.getAction();

Bundle b = intent.getExtras();

Object[] lstName = b.keySet().toArray();

// 显示所有收到的消息及其细节

for (int i = 0; i < lstName.length; i++) {

String keyName = lstName[i].toString();

Log.e(keyName, String.valueOf(b.get(keyName)));

}

//搜索设备时,取得设备的MAC地址

if (BluetoothDevice.ACTION_FOUND.equals(action)) {

BluetoothDevice device = intent

.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);

String str= device.getName() + "|" + device.getAddress();

if (lstDevices.indexOf(str) == -1)// 防止重复添加

lstDevices.add(str); // 获取设备名称和mac地址

adtDevices.notifyDataSetChanged();

}

}

};

2.使用BlueAdatper的搜索:

btAdapt.startDiscovery();
    3.在BroadcastReceiver的onReceive()里取得搜索所得的蓝牙设备信息(如名称,MAC,RSSI);
    4.通过设备的MAC地址来建立一个BluetoothDevice对象;

5.由BluetoothDevice衍生出BluetoothSocket,准备SOCKET来读写设备;

6.通过BluetoothSocket的createRfcommSocketToServiceRecord()方法来选择连接的协议/服务,这里用的       是SPP(UUID:00001101-0000-1000-8000-00805F9B34FB);

try {

btSocket = btDev.createRfcommSocketToServiceRecord(uuid);

catch (IOException e) {

// TODO Auto-generated catch block

Log.e(TAG, "Low: Connection failed.", e);

}

成功后进行连接:

try {

btSocket.connect();

Log.e(TAG, " BT connection established, data transfer link open.");

mangeConnectedSocket(btSocket);//自定义函数进行蓝牙通信处理

catch (IOException e) {

Log.e(TAG, " Connection failed.", e);

setTitle("连接失败..");

}

7.Connect之后(如果还没配对则系统自动提示),使用

BluetoothSocket的getInputStream()和getOutputStream()来读写蓝牙设备。

 读写可以归到一个独立线程去实现~ 注意:读时必须一直循环读取串口缓冲区,写可以不需要。

按以上7步逐次走过后,你就会发现Android蓝牙模块是多么的坑爹了。

出现问题:

在第6步一般初学者都会报错: 执行.connect()发生异常Connection refused

此时执行不下去咯,怎么办怎么办呢?

于是边debug边网上找攻略,总算在Google出老外的一些做法,尝试了下,貌似还可行。也即把

btSocket的建立方法采用另一种方法替代,这里都使用端口1

Method m;

try {

m = btDev.getClass().getMethod("createRfcommSocket", new Class[] {int.class});

    btSocket = (BluetoothSocket) m.invoke(btDev, Integer.valueOf(1));

catch (SecurityException e1) {

// TODO Auto-generated catch block

e1.printStackTrace();

catch (NoSuchMethodException e1) {

// TODO Auto-generated catch block

e1.printStackTrace();

catch (IllegalArgumentException e) {

// TODO Auto-generated catch block

e.printStackTrace();

catch (IllegalAccessException e) {

// TODO Auto-generated catch block

e.printStackTrace();

catch (InvocationTargetException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

至此,这个问题貌似倒也解决了,程序继续往下跑。

但这里请记住之前的异常,先别急着抛开~人家不一定一直都是异常哦

接下来的任务是,让手机通过蓝牙跟单片机的蓝牙模块通信,并发送数据,通过电脑串口调试助手显示出来。具体实现,在mangeConnectedSocket(btSocket)方法中实现,里面通过启动另一个Activity实现。不是重点,略过。

直到这里,我们都只是把手机蓝牙模块充当客户端来使用,那什么时候会用到服务端呢?其实,之前手机蓝牙与单片机蓝牙模块的通信,单片机蓝牙模块就充当了服务端(处于监听状态,被手机蓝牙连接)。为了更好地搞清楚Android蓝牙通信,我们接下来使用2个手机的蓝牙进行通信。简单地说,就是做一个“手机蓝牙扣扣”,⊙﹏⊙b汗

一开始就想天真地把之前的程序同时烧到2部手机中,发现只有一部手机能正常建立socket连接(主动连接的那台),而另一部却迟迟没有响应。原因很简单,服务端的程序还没有编写!

于是,开始服务端程序:开辟一个新的线程实现

连接蓝牙设备——蓝牙服务端:

    class AcceptThread extends Thread {

private final BluetoothServerSocket serverSocket;

public AcceptThread() {

// Use a temporary object that is later assigned to mmServerSocket,

// because mmServerSocket is final

BluetoothServerSocket tmp=null;

try {

//tmp = btAdapt.listenUsingRfcommWithServiceRecord("MyBluetoothApp", uuid);

Log.e(TAG, "++BluetoothServerSocket established!++");

Method listenMethod =        btAdapt.getClass().getMethod("listenUsingRfcommOn",

new   Class[]{int.class});

tmp = ( BluetoothServerSocket) listenMethod.invoke(btAdapt,                                                  Integer.valueOf( 1));

catch (SecurityException e) {

// TODO Auto-generated catch block

e.printStackTrace();

catch (IllegalArgumentException e) {

// TODO Auto-generated catch block

e.printStackTrace();

catch (NoSuchMethodException e) {

// TODO Auto-generated catch block

e.printStackTrace();

catch (IllegalAccessException e) {

// TODO Auto-generated catch block

e.printStackTrace();

catch (InvocationTargetException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

serverSocket=tmp;

}

public void run() {

// Keep listening until exception occurs or a socket is returned

//mState!=STATE_CONNECTED

while(true) {//这里是一直循环监听,也可以设置mState来判断

try {

socket = serverSocket.accept();

Log.e(TAG, "++BluetoothSocket established! DataLink open.++");

catch (IOException e) {

break;

}

// If a connection was accepted

if (socket != null) {

// Do work to manage the connection (in a separate thread)

manageConnectedSocket();

try {

serverSocket.close();

catch (IOException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

break;

}

}

}

public void cancel() {

try {

serverSocket.close();

catch (IOException e) { }

}

}

安装测试:当2部手机都装上并打开同样的程序后,通过蓝牙检索并连接,经测试可以成功连接上,双双进入“聊天界面”,嘿嘿

注意,这时候重新拾回之前那个异常,把socket连接建立的方法重新改为

btSocket = btDev.createRfcommSocketToServiceRecord(uuid);//客户端

对应的服务端程序:

tmp = btAdapt.listenUsingRfcommWithServiceRecord("MyBluetoothApp", uuid);//服务端

这样继续重新运行安装测试,在2部手机上运行发现之前那个bug消失了~2部手机又双双进入聊天界面。

神奇~

存在bug

任一一部手机都只能成功启动一次作为客户端的主动连接,当退出聊天界面回到主界面时(服务端的AcceptThread还在继续运行着),可再次主动连接另一部手机时就又报异常Connection refused。也就是说 客户端的蓝牙套接字2次连接时出错~哎(注意我的客户端蓝牙连接程序是没有放到一个独立线程,而是放到一个按钮监听事件中)

又折腾了好久,没发现个所以然来,看来连完一次退出再连时就只好重启程序咯。有哪位大神知道为什么的麻烦告知下哈!

若需要代码,code下载

Android蓝牙串口通讯【转】的更多相关文章

  1. android 蓝牙串口通讯使用简介

    需要的权限 <uses-permission android:name="android.permission.BLUETOOTH" />  <uses-perm ...

  2. Android蓝牙串口程序开发

    本文主要介绍了针对android的蓝牙串口上位机开发. 程序下载地址:点击打开链接 一.帧定义 androidclient依照一定的数据帧格式通过蓝牙串口发送数据到连接到MCU的蓝牙从机.MCU接收到 ...

  3. 手机与Arduino蓝牙串口通讯实验及完整例程

    安卓手机与Arduino之间采用蓝牙串口通讯,是很多智能装置和互动装置常用的控制方法,简单而有效,无需网络环境,很实用的技术. 实验采用Arduino UNO板,加了一块1602LCD屏做显示(因为只 ...

  4. Android 蓝牙串口通信工具类 SerialPortUtil 3.0.+

    建议使用4.+版本,避免一些不必要的bug.4.+版本文档地址:https://www.cnblogs.com/shanya/articles/16062256.html SerialPortUtil ...

  5. Android(十六 ) android 与蓝牙串口通讯

    1.得到蓝牙适配器 BluetoothAdapter mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter(); 2.打开蓝牙 if (!mBl ...

  6. Android端简易蓝牙聊天通讯App(原创)

    欢迎转载,但请注明出处!谢谢.http://www.cnblogs.com/weizhxa/p/5792775.html 最近公司在做一个蓝牙串口通讯的App,有一个固定的蓝牙设备,需要实现手机连接相 ...

  7. HC-07 蓝牙串口模块

    http://www.wavesen.com/probig.asp?id=17 本模块为新推出的产品,各项功能和性能.及引脚封装,均兼容于HC-06. 为低成本需求的的客户推荐本产品.相比HC-06来 ...

  8. Android通过JNI实现与C语言的串口通讯操作蓝牙硬件模块

    一直想写一份技术文档,但因为自感能力有限而无从下笔,近期做了个关于Android平台下实现与C语言的通讯来操作蓝牙模块的项目,中间碰到了很多问题,也在网上查了很多资料,在完毕主要功能后.也有一些人在网 ...

  9. Android串口通讯

    今天在整一个项目,需要利用串口通讯在网上看了好多人的帖子才稍微整出了一点头绪. 首先串口代码就是利用谷歌自己的api,将java代码放在java/android_serialport_api目录下,如 ...

随机推荐

  1. 面向对象-Object类

    一.Object类中的equals()方法 equals(Object obj) :指示其它某个对象是否与此对象"相等". 返回值类型是boolean Oblect类中的equal ...

  2. Yandex.Algorithm 2011 A. Double Cola

    1.题目描写叙述:点击打开链接 2.解题思路:本题是一道找规律的数学题,通过题意描写叙述不难知道,相当于有5棵二叉树构成了一个森林,须要你按层次遍历找到第n个人是谁. 观察后不难发现,如果最開始的一层 ...

  3. 网络相关系列之四:数据解析之SAX方式解析XML数据

    一.XML和Json数据的引入: 通常情况下.每一个须要訪问网络的应用程序都会有一个自己的server.我们能够向server提交数据,也能够从server获取数据.只是这个时候就有一个问题,这些数据 ...

  4. hdu 5881 Tea (2016 acm 青岛网络赛)

    原题地址:http://acm.hdu.edu.cn/showproblem.php?pid=5881 Tea Time Limit: 3000/1000 MS (Java/Others)    Me ...

  5. Lumen开发:结合Redis实现消息队列(2)

    上一篇讲了Lumen配置Redis,现在来讲一下,如何实现消息队列 2.编写任务类 2.1  任务类结构 默认情况下,应用的所有队列任务都存放在app/Jobs目录.任务类非常简单,正常情况下只包含一 ...

  6. 安装virtualBox 增强包

    1 在原始操作系统安装. 2 打开USB设置. 3 运行虚拟机中的Linux中,Device->install guest additions 再安装增强包. 4 插入U盘,如果这时可以看到U盘 ...

  7. 从零开始写一个Exporter

    前言 上一篇文章中已经给大家整体的介绍了开源监控系统Prometheus,其中Exporter作为整个系统的Agent端,通过HTTP接口暴露需要监控的数据.那么如何将用户指标通过Exporter的形 ...

  8. python 快速排序详述

    快速排序是对“冒泡排序”的优化算法,都属于交换排序类. 描述:它通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据要小,然后再按此方法对这两部分数据分别进行快速 ...

  9. windowsphone8.1学习笔记之位图编程

    说位图,先把image控件简单过下,Image的Source设置 <Image Name="img" Source="可以是网络图片的Uri.应用文件的Uri或者安 ...

  10. Centos设置开机启动Apache和Mysql

    先用chkconfig --list查询apache和mysql服务是否存在,不存在则需要手动添加 [root@centos64 vsftpd]# chkconfig --list 测试存在,只需要开 ...