Android 蓝牙
添加权限:
<uses-permission Android:name="android.permission.BLUETOOTH_ADMIN"/>
<uses-permission android:name="android.permission.BLUETOOTH"/>
- 客户端
开启蓝牙:
/**
* 打开蓝牙设备
*/
void openBT(){
mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
if (mBluetoothAdapter!= null){
if (!mBluetoothAdapter.isEnabled()){
mBluetoothAdapter.enable();
Log.d("qfopenBT","打开蓝牙成功");
}
}
}
搜索蓝牙:
/**
* 搜索蓝牙设备
* @param view
*/
@OnClick(R.id.btSearch)
public void startSearch(View view){
if(mBluetoothAdapter!= null &&mBluetoothAdapter.isEnabled()){
if (!mBluetoothAdapter.isDiscovering()){
mBluetoothAdapter.startDiscovery();
}
}
}
开启搜索是异步操作,发现设备后会发送广播,所以要定义广播接收者
在接收到广播后,获取广播里的蓝牙数据
private class BTBroadCastRevextends BroadcastReceiver{
@Override
public void onReceive(Context context,Intent intent) {
String strAction = intent.getAction();
if (strAction.equals(BluetoothDevice.ACTION_FOUND)){
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
mArrDevice.add(device);
mAdapter.notifyDataSetChanged();
}
else if(strAction.equals(BluetoothAdapter.ACTION_DISCOVERY_FINISHED)){
Log.d("qfonReceive","搜索完成");
}
}
}
//注册广播接收者
myReceive = newBTBroadCastRev();
IntentFilter ifFind = new IntentFilter(BluetoothDevice.ACTION_FOUND);
this.registerReceiver(myReceive,ifFind);
IntentFilter ifFinishFind = newIntentFilter(BluetoothAdapter.ACTION_DISCOVERY_FINISHED);
this.registerReceiver(myReceive,ifFinishFind);
连接蓝牙,并开启发送数据线程:
/**
* 点击item,连接对应的蓝牙设备
*/
protected void connectBT() {
mLvDevice.setOnItemClickListener(newAdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent,View view, int position, longid) {
MyClientTask task = newMyClientTask();
task.execute(mArrDevice.get(position));
}
});
}
class MyClientTask extends AsyncTask<BluetoothDevice,Void,Void>{
@Override
protected VoiddoInBackground(BluetoothDevice... devices) {
BluetoothDevice device = devices[0];
try {
//使用安全连接,服务端也要一样使用安全连接,UUID也要跟服务器的监听UUID一致
BluetoothSocket socket = device.createRfcommSocketToServiceRecord(MY_UUID_SECURE);
socket.connect();
Log.d("qfdoInBackground_client","连接成功,开始发送数据");
byte[] btMsg =new String("Hello").getBytes();
socket.getOutputStream().write(btMsg,0,btMsg.length);
} catch(Exception e) {
e.printStackTrace();
}
return null;
}
}
uuid可以通过uuidgen生成,生成结果类似以下结构:
5D3D5E52-338A-47B8-9F10-27ADF89E204E
- 服务端
开启蓝牙,跟客户端一样
启动服务线程
new RevTask().execute();
class RevTask extendsAsyncTask<Void,Void,String>{
@Override
protected void onPostExecute(String s) {
super.onPostExecute(s);
tvMsg.setText(s);
}
@Override
protected StringdoInBackground(Void... params) {
try {
Log.d("qfdoInBackground","开始监听");
//要跟客户端uuid一致
BluetoothServerSocket sevSocket =mBluetoothAdapter.listenUsingRfcommWithServiceRecord("blue_service",MY_UUID_SECURE);
BluetoothSocket socket = sevSocket.accept();
if(socket != null){
Log.d("qfdoInBackground","连接成功");
InputStream stream = socket.getInputStream();
byte[] btRead =new byte[1024];
int iLength = stream.read(btRead);
Log.d("qfdoInBackground","读取数据成功"+iLength);
String strMsg =new String(btRead,"utf-8");
Log.d("qfdoInBackground",strMsg);
return strMsg;
}
else{
Log.d("qfdoInBackground","失败");
}
} catch (IOException e) {
e.printStackTrace();
Log.d("qfdoInBackground","异常");
}
return null;
}
}
上述代码没有实现配对,对应经典蓝牙通信,最好先进行配对再连接,已经配对的蓝牙设备可以直接通过adapter获取到
//得到所有已经配对的蓝牙适配器对象
Set<BluetoothDevice> devices = adapter.getBondedDevices();
没有配对的蓝牙设备,可以在扫描设备的广播通知中判断:
if
(device.getBondState() != BluetoothDevice.BOND_BONDED) { ...... }点击设备连接时,判断是否已经配对,如果已经配对,直接连接,如果没有配对,先配对:
- if (btDev.getBondState() == BluetoothDevice.BOND_NONE) {
- //利用反射方法调用BluetoothDevice.createBond(BluetoothDevice remoteDevice);
- Method createBondMethod = BluetoothDevice.class
- .getMethod("createBond");
- Log.d("BlueToothTestActivity", "开始配对");
- returnValue = (Boolean) createBondMethod.invoke(btDev);
- }else if(btDev.getBondState() == BluetoothDevice.BOND_BONDED){
- connect(btDev);
- }
配对结果也会通过广播传递结果信息:
- // 注册Receiver来获取蓝牙设备相关的结果
- IntentFilter intent = new IntentFilter();
- intent.addAction(BluetoothDevice.ACTION_FOUND);// 用BroadcastReceiver来取得搜索结果
- intent.addAction(BluetoothDevice.ACTION_BOND_STATE_CHANGED);
- intent.addAction(BluetoothAdapter.ACTION_SCAN_MODE_CHANGED);
- intent.addAction(BluetoothAdapter.ACTION_STATE_CHANGED);
- registerReceiver(searchDevices, intent);
- if(BluetoothDevice.ACTION_BOND_STATE_CHANGED.equals(action)){
- device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
- switch (device.getBondState()) {
- case BluetoothDevice.BOND_BONDING:
- Log.d("BlueToothTestActivity", "正在配对......");
- break;
- case BluetoothDevice.BOND_BONDED:
- Log.d("BlueToothTestActivity", "完成配对");
- connect(device);//连接设备
- break;
- case BluetoothDevice.BOND_NONE:
- Log.d("BlueToothTestActivity", "取消配对");
- default:
- break;
- }
Android 蓝牙的更多相关文章
- android蓝牙打印机
您还未登录!|登录|注册|帮助 首页 业界 移动 云计算 研发 论坛 博客 下载 更多 reality_jie的专栏 编程的过程是一种微妙的享受 目录视图 摘要视图 订阅 CSDN2013 ...
- Android蓝牙实例(和单片机蓝牙模块通信)
最近做毕设,需要写一个简单的蓝牙APP进行交互,在网上也找了很多资料,终于给搞定了,这里分享一下^_^. 1.Android蓝牙编程 蓝牙3.0及以下版本编程需要使用UUID,UUID是通用唯一识别码 ...
- Android 蓝牙4.0 BLE
Android ble (Bluetooth Low Energy) 蓝牙4.0,也就是说API level >= 18,且支持蓝牙4.0的手机才可以使用. BLE是蓝牙4.0的核心Profil ...
- android 蓝牙4.0 开发介绍
最近一直在研究一个蓝牙功能 由于本人是菜鸟 学起来比较忙 一直搞了好久才弄懂 , 网上对蓝牙4.0也就是几个个dome 抄来抄去,全是英文注解 , 对英语不好的朋友来说 真是硬伤 , 一些没必要的描 ...
- 【转】android蓝牙开发---与蓝牙模块进行通信--不错
原文网址:http://www.cnblogs.com/wenjiang/p/3200138.html 近半个月来一直在搞android蓝牙这方面,主要是项目需要与蓝牙模块进行通信.开头的进展很顺利, ...
- Android 蓝牙开发(整理大全)
Android蓝牙开发 鉴于国内Android蓝牙开发的例子很少,以及蓝牙开发也比较少用到,所以找的资料不是很全. (一): 由于Android蓝牙的通信都需要用到UUID,如果由手机发起搜索,当搜索 ...
- android -- 蓝牙 bluetooth (四)OPP文件传输
在前面android -- 蓝牙 bluetooth (一) 入门文章结尾中提到了会按四个方面来写这系列的文章,前面已写了蓝牙打开和蓝牙搜索,这次一起来看下蓝牙文件分享的流程,也就是蓝牙应用opp目录 ...
- android -- 蓝牙 bluetooth (三)搜索蓝牙
接上篇打开蓝牙继续,来一起看下蓝牙搜索的流程,触发蓝牙搜索的条件形式上有两种,一是在蓝牙设置界面开启蓝牙会直接开始搜索,另一个是先打开蓝牙开关在进入蓝牙设置界面也会触发搜索,也可能还有其它触发方式,但 ...
- android -- 蓝牙 bluetooth (一) 入门
前段时间在 网上看了一些关于android蓝牙的文章,发现大部分是基于老版本(4.1以前含4.1)的源码,虽然无碍了解蓝牙的基本原理和工作流程,但对着4.2.2的代码看起来总是有些遗憾.所以针对4.2 ...
- 深入了解Android蓝牙Bluetooth——《基础篇》
什么是蓝牙? 也可以说是蓝牙技术.所谓蓝牙(Bluetooth)技术,实际上是一种短距离无线电技术,是由爱立信公司公司发明的.利用"蓝牙"技术,能够有效地简化掌上电脑.笔记本电 ...
随机推荐
- Xcode清除缓存等
Xcode出现一些错误的时候,有时候不是代码的问题,需要清理一下Xcode的缓存或者项目的Product等: 1. Product清理 1.1 Product-Clean 1.2 Product- ...
- jquery easyui tree的全选与反选
//全选反选 //参数:selected:传入this,表示当前点击的组件 //treeMenu:要操作的tree的id:如:id="userTree" function tree ...
- QT socket相关
#include<QtNetwork/QTcpSocket>#include<QtNetwork/QTcpServer> 1.服务器端 void About::init_tcp ...
- WebView 一直展示加载中。。。
webview加载html5页面总是一直在加载中,加载很慢或干脆加载不出来, 听浏览器的大牛说可能是 js导致的,尝试在onpause里加入mWebView.pauseTimers(), onResu ...
- Python Set Literals
现有3种方式创建set() >>> def f(): ... return set([1, 2, 3]) ... >>> def h(): ... return s ...
- Android开源图表之树状图和饼状图的官方示例的整理
最近由于工作需要,所以就在github上搜了下关于chart的三方框架 官方地址https://github.com/PhilJay/MPAndroidChart 由于工作需要我这里整理了一份Ecli ...
- JSON.stringify////////////////////////////////zzzzzzzzzzzzzz
JSON.stringify 语法实例讲解 可能有些人对系列化这个词过敏,我的理解很简单.就是说把原来是对象的类型转换成字符串类型(或者更确切的说是json类型的).就这么简单.打个比方说,你有一个类 ...
- 在虚拟机中配置FastDFS+Nginx模块
先上部署图 提示一下, ip 192.168.72.138 上面部署了两个group, 分别为 group1和g2. 另外, 同组之内的 port 要保持一致. 一.安装准备 1. #每台机器都添加两 ...
- Timer的故事----Jdk源码解读
咱们今天也来说说定时器Timer Timer是什么? Timer n. [电子] 定时器:计时器:计时员 从翻译来看,我们可以知道Timer的本意是,定时定点. 而JDK中Timer类也的确是这个本 ...
- Visual Studio配色方案
Eclipse开源工具和VS在诸多方面真的是差距非常大,无奈Java编程,使用VS非常麻烦.所以只能选择Eclipse 但是Eclipse的系统配色,又实在是不舒服,于是抽时间,从VS上抠了一份默认的 ...