基于Android的BLE通信软件

2023-10-17 19:38
文章标签 android ble 通信软件

本文主要是介绍基于Android的BLE通信软件,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到教程。
人工智能编程入门博客

实现目标

自己编写基于Qt的Android软件,用于实现手机与TB-02-kit模块进行数据通讯。

Android软件发送的数据,经TB-02-kit模块转发至串口助手中输出;

串口助手发送的数据可以在Android软件中显示,进而实现BLE的数据双向通信。

所需工具及环境

  • TB-02-kit模块
  • Qt Creator 4.10.1
  • Qt 5.13.1
  • XCOM V2.0 串口助手
  • Android 手机
  • 本人电脑 Windows 10 64bit [版本 10.0.19041.329]

相关阅读 前置知识

给大家介绍一款好用的蓝牙BT5.0透传模块
Windows下基于Qt开发Android应用
BLE中这些概念你都了解吗

本文源码

因为是第一次分享Qt代码,为了方便大家学习,代码添加了大量注释。
后台回复关键字“Android-BLE”,获取本文涉及到的软件及Qt工程源码。

具体实现

1. 要使用qt蓝牙模块, 项目的 .pro要声明使用


2. 扫描设备

在构造函数中即执行蓝牙设备扫描,即软件一启动就执行扫描。

Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);//创建搜索服务:https://doc.qt.io/qt-5/qbluetoothdevicediscoveryagent.htmldiscoveryAgent =new QBluetoothDeviceDiscoveryAgent(this);//设置BLE的搜索时间discoveryAgent->setLowEnergyDiscoveryTimeout(20000);connect(discoveryAgent,SIGNAL(deviceDiscovered(QBluetoothDeviceInfo)),this,SLOT(addBlueToothDevicesToList(QBluetoothDeviceInfo)));//找到设备之后添加到列表显示出来connect(discoveryAgent, SIGNAL(finished()), this, SLOT(scanFinished()));connect(discoveryAgent, SIGNAL(canceled()), this, SLOT(scanCanceled()));connect(this, SIGNAL(returnAddress(QBluetoothDeviceInfo)), this, SLOT(createCtl(QBluetoothDeviceInfo)));//开始进行设备搜索discoveryAgent->start(QBluetoothDeviceDiscoveryAgent::LowEnergyMethod);
}
3. 将扫描结果添加到QListWidget中
//deviceDiscovered signals 对应的槽函数
void Widget::addBlueToothDevicesToList(const QBluetoothDeviceInfo &info)
{if (info.coreConfigurations() & QBluetoothDeviceInfo::LowEnergyCoreConfiguration)           //获取设备信息,并判断该设备是否为BLE设备{//格式化设备地址和设备名称QString label = QString("%1 %2").arg(info.address().toString()).arg(info.name());//检查设备是否已存在,避免重复添加QList<QListWidgetItem *> items = ui->ctrBleList->findItems(label, Qt::MatchExactly);//不存在则添加至设备列表if (items.empty()){QListWidgetItem *item = new QListWidgetItem(label);ui->ctrBleList->addItem(item);devicesList.append(info);}}
}
4. 连接蓝牙,停止扫描
void Widget::on_btnConnectBle_clicked()
{//确认选取了某一个蓝牙设备if(!ui->ctrBleList->currentItem()->text().isEmpty()){//获取选择的地址QString bltAddress = ui->ctrBleList->currentItem()->text().left(17);for (int i = 0; i<devicesList.count(); i++){//地址对比if(devicesList.at(i).address().toString().left(17) == bltAddress){QBluetoothDeviceInfo choosenDevice = devicesList.at(i);//发送自定义signals==>执行slots:createCtlemit returnAddress(choosenDevice);//停止搜索服务discoveryAgent->stop();break;}}}
}
5. 获取特征
void Widget::searchCharacteristic()
{if(m_bleServer){QList<QLowEnergyCharacteristic> list=m_bleServer->characteristics();qDebug()<<"[xiaohage]list.count()="<<list.count();//遍历characteristicsfor(int i=0;i<list.count();i++){QLowEnergyCharacteristic c=list.at(i);/*如果QLowEnergyCharacteristic对象有效,则返回true,否则返回false*/if(c.isValid()){//返回特征的属性。//这些属性定义了特征的访问权限。if(c.properties() & QLowEnergyCharacteristic::WriteNoResponse || c.properties() & QLowEnergyCharacteristic::Write){ui->ctrSystemLogInfo->insertPlainText("\n具有写权限!");m_writeCharacteristic = c;  //保存写权限特性if(c.properties() & QLowEnergyCharacteristic::WriteNoResponse){m_writeMode = QLowEnergyService::WriteWithoutResponse;}else{m_writeMode = QLowEnergyService::WriteWithResponse;}}if(c.properties() & QLowEnergyCharacteristic::Read){m_readCharacteristic = c; //保存读权限特性}//描述符定义特征如何由特定客户端配置。m_notificationDesc = c.descriptor(QBluetoothUuid::ClientCharacteristicConfiguration);//值为真if(m_notificationDesc.isValid()){//写描述符m_bleServer->writeDescriptor(m_notificationDesc, QByteArray::fromHex("0100"));ui->ctrSystemLogInfo->insertPlainText("\n写描述符!");}}}}
}
6. 发送数据

writeCharacteristic()方法,发送数据给ble设备。
点击界面中的"发送"按钮,发送"Hello World"字符串。

void Widget::SendMsg(QString text)
{QByteArray array=text.toLocal8Bit();m_bleServer->writeCharacteristic(m_writeCharacteristic,array, m_writeMode);
}void Widget::on_btnSendData_clicked()
{SendMsg("Hello World");
}
7. 写入数据

通过蓝牙QLowEnergyService::characteristicRead的回调接口,接收蓝牙收的消息。

void Widget::BleServiceCharacteristicRead(const QLowEnergyCharacteristic &c,const QByteArray &value)
{Q_UNUSED(c)ui->ctrSystemLogInfo->insertPlainText("\n当特征读取请求成功返回其值时:");ui->ctrSystemLogInfo->insertPlainText(QString(value));
}
8. 断开连接
Widget::~Widget()
{if(!(m_BLEController->state() == QLowEnergyController::UnconnectedState))m_BLEController->disconnectFromDevice();//从设备断开链接delete ui;
}

界面布局

结果展示

如果出现" Cannot connect to remote device. " ,可以点击"连接"按钮重新连接一下。

串口助手及应用程序输出

To do

本实例只是演示一下Android手机与TB-02-kit模块的通讯过程,程序里有需要完善的地方,比如,应该增加一个"扫描"按钮,而不是软件启动过程中直接进行蓝牙扫描,这样的话,就需要蓝牙的上电要在软件启动之前完成。

程序的健壮性也要完善,比如偶尔会出现与模块无法正常连接的情况,需要再次点击"连接"按钮才可,这些工作你们自己可以完善一下哈。

有了本部分知识,下一步我们结合Android手机和TB-02-kit模块,实现STM32的设备的远程控制。

Qt小知识

1. Qt Creator程序输出窗口过滤调试信息

2. 为Button添加事件

Button控件右键菜单中选中“转到槽…”,然后在弹出列表中选中信号:“clicked() ”,然后点击OK按钮,即可进入其事件函数中。

参考资料

Qt官方文档:https://doc.qt.io/qt-5/classes.html

欢迎关注

程序员小哈带你玩转嵌入式,微信搜索:嵌入式从0到1,更多干货等着你。

这篇关于基于Android的BLE通信软件的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/227503

相关文章

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

android-opencv-jni

//------------------start opencv--------------------@Override public void onResume(){ super.onResume(); //通过OpenCV引擎服务加载并初始化OpenCV类库,所谓OpenCV引擎服务即是 //OpenCV_2.4.3.2_Manager_2.4_*.apk程序包,存

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动

Android 10.0 mtk平板camera2横屏预览旋转90度横屏拍照图片旋转90度功能实现

1.前言 在10.0的系统rom定制化开发中,在进行一些平板等默认横屏的设备开发的过程中,需要在进入camera2的 时候,默认预览图像也是需要横屏显示的,在上一篇已经实现了横屏预览功能,然后发现横屏预览后,拍照保存的图片 依然是竖屏的,所以说同样需要将图片也保存为横屏图标了,所以就需要看下mtk的camera2的相关横屏保存图片功能, 如何实现实现横屏保存图片功能 如图所示: 2.mtk

android应用中res目录说明

Android应用的res目录是一个特殊的项目,该项目里存放了Android应用所用的全部资源,包括图片、字符串、颜色、尺寸、样式等,类似于web开发中的public目录,js、css、image、style。。。。 Android按照约定,将不同的资源放在不同的文件夹中,这样可以方便的让AAPT(即Android Asset Packaging Tool , 在SDK的build-tools目

Android fill_parent、match_parent、wrap_content三者的作用及区别

这三个属性都是用来适应视图的水平或者垂直大小,以视图的内容或尺寸为基础的布局,比精确的指定视图的范围更加方便。 1、fill_parent 设置一个视图的布局为fill_parent将强制性的使视图扩展至它父元素的大小 2、match_parent 和fill_parent一样,从字面上的意思match_parent更贴切一些,于是从2.2开始,两个属性都可以使用,但2.3版本以后的建议使

Android Environment 获取的路径问题

1. 以获取 /System 路径为例 /*** Return root of the "system" partition holding the core Android OS.* Always present and mounted read-only.*/public static @NonNull File getRootDirectory() {return DIR_ANDR

Android逆向(反调,脱壳,过ssl证书脚本)

文章目录 总结 基础Android基础工具 定位关键代码页面activity定位数据包参数定位堆栈追踪 编写反调脱壳好用的脚本过ssl证书校验抓包反调的脚本打印堆栈bilibili反调的脚本 总结 暑假做了两个月的Android逆向,记录一下自己学到的东西。对于app渗透有了一些思路。 这两个月主要做的是代码分析,对于分析完后的持久化等没有学习。主要是如何反编译源码,如何找到

android系统源码12 修改默认桌面壁纸--SRO方式

1、aosp12修改默认桌面壁纸 代码路径 :frameworks\base\core\res\res\drawable-nodpi 替换成自己的图片即可,不过需要覆盖所有目录下的图片。 由于是静态修改,则需要make一下,重新编译。 2、方法二Overlay方式 由于上述方法有很大缺点,修改多了之后容易遗忘自己修改哪些文件,为此我们采用另外一种方法,使用Overlay方式。