当前位置: 首页 > news >正文

esp32c3开发板通过micropython的ubluetooth库连蓝牙设备

ESP32-C3开发板是一款高性能、低功耗的微控制器,搭载了Espressif自家的RISC-V处理器。通过MicroPython,一种面向微控制器的精简版Python编程语言,开发者可以轻松地为ESP32-C3编写代码。MicroPython的ubluetooth库使得ESP32-C3能够通过蓝牙与各种设备进行通信。

使用ubluetooth库,ESP32-C3可以作为蓝牙主机或从机,实现与蓝牙设备的配对、连接和数据传输。开发者可以利用这个库来创建蓝牙低功耗(BLE)应用,如智能家居设备、健康监测设备等。通过简单的Python代码,ESP32-C3可以发送和接收蓝牙信号,实现设备间的无线通信。这使得ESP32-C3成为一个多功能、易于编程的蓝牙解决方案,适合各种物联网项目。

首先学习一下蓝牙通讯的机制

蓝牙通讯介绍

蓝牙通信是一种无线通信技术,用于在短距离内(通常在10米以内)连接不同的电子设备,如手机、平板电脑、笔记本电脑、耳机、扬声器等。蓝牙通信的机制主要包括以下几个方面:

1. 频率和调制

蓝牙通信使用2.4 GHz的工业、科学和医疗(ISM)频段。为了避免与其他无线设备的干扰,蓝牙采用了频率跳跃扩频(FHSS)技术。在通信过程中,蓝牙设备会在79个不同的频率通道上进行跳变,每秒跳变1600次,从而减少干扰和提高通信的稳定性。

2. 设备配对和连接

蓝牙设备之间的通信需要先进行配对和连接:

  • 配对:两个设备通过交换密钥建立信任关系。配对过程通常需要用户确认,以确保安全性。
  • 连接:配对成功后,设备可以建立稳定的连接,进行数据传输。

3. 主从设备

蓝牙通信采用主从模式:

  • 主设备:控制连接和数据传输的设备,可以同时连接多个从设备。
  • 从设备:被主设备控制的设备,通常只能连接一个主设备。

4. 服务和特征

蓝牙低功耗(BLE)协议中,设备通过服务和特征来交换数据:

  • 服务:一组相关的特征集合,代表设备的某种功能或数据类型。
  • 特征:具体的数据点,可以被读取、写入或通知。

5. 广播和扫描

  • 广播:设备通过广播包发送信息,其他设备可以通过扫描接收这些信息。
  • 扫描:设备监听周围的广播包,以发现可连接的蓝牙设备和服务。

6. 数据传输协议

蓝牙通信使用多种协议层来确保数据的可靠传输:

  • 物理层(PHY):负责实际的无线信号传输。
  • 链路层(LL):管理设备的连接状态、数据包的发送和接收。
  • 主机控制接口(HCI):提供主机和控制器之间的通信接口。
  • 逻辑链路控制和适配协议(L2CAP):负责数据的分段、重组和错误控制。
  • 属性协议(ATT):在BLE中用于服务的发现和数据的读写操作。

7. 安全机制

蓝牙通信包括多种安全措施:

  • 加密:使用AES加密算法保护数据传输。
  • 认证:通过配对过程中的PIN码或OOB(Out of Band)数据进行设备身份验证。
  • 授权:确保只有经过授权的设备才能访问特定的服务和特征。

8. 功耗管理

特别是对于BLE,功耗管理是一个关键因素:

  • 低功耗模式:设备可以在不需要通信时进入休眠状态,减少能量消耗。
  • 连接间隔:调整主从设备之间的连接间隔,平衡数据传输速率和能耗。

总之,蓝牙通信通过一系列复杂的协议和技术,实现了设备间的短距离、低功耗、安全的无线数据传输。

实践

准备好esp32c3开发板,进入micropython交互环境,输入如下代码:

蓝牙测试代码1

import ubluetooth
import utime
import machineclass BLEServer:def __init__(self):self.ble = ubluetooth.BLE()self.ble.active(True)self.ble.irq(self.ble_irq)self.register_service()self.connected = Falsedef register_service(self):# 创建一个UUID为"12345678-1234-5678-1234-56789abcdef0"的服务service_uuid = ubluetooth.UUID('12345678-1234-5678-1234-56789abcdef0')service = self.ble.gatts_register_services([(service_uuid, ((ubluetooth.UUID('12345678-1234-5678-1234-56789abcdef1'), ubluetooth.FLAG_READ),))])self.service_handle = service[0]self.char_handle = service[1]# 设置Characteristic的初始值self.ble.gatts_write(self.char_handle, b'Hello, BLE!')def ble_irq(self, event, data):if event == 1:# Central设备连接self.connected = Trueprint("Device connected")elif event == 2:# Central设备断开连接self.connected = Falseprint("Device disconnected")# 重置广告,以便再次被发现self.ble.gap_advertise(100, adv_data=self.ble.config('mac') + b'\x02\x01\x06' + b'\x11\x07' + ubluetooth.UUID('12345678-1234-5678-1234-56789abcdef0').bytes)def start_advertising(self):# 开始广告,以便Central设备可以发现这个Peripheralself.ble.gap_advertise(100, adv_data=self.ble.config('mac') + b'\x02\x01\x06' + b'\x11\x07' + ubluetooth.UUID('12345678-1234-5678-1234-56789abcdef0').bytes)def main():ble_server = BLEServer()ble_server.start_advertising()while True:if ble_server.connected:# 可以在这里添加逻辑来处理连接状态passutime.sleep(1)if __name__ == "__main__":main()

但是这段代码有问题,

蓝牙测试代码2

后来用了这段文心生成的代码:

# 文心的例子
import ubluetooth
import machine
import time# 初始化蓝牙
ble = ubluetooth.BLE()
ble.active(True)# 定义服务UUID
# SERVICE_UUID = '12345678-1234-5678-1234-56789abcdef0'
# CHARACTERISTIC_UUID = '12345678-1234-5678-1234-56789abcdef1'# 创建BLE服务和特性
def ble_init():# services = (#     (#         ubluetooth.UUID(SERVICE_UUID),#         (#             (ubluetooth.UUID(CHARACTERISTIC_UUID), ubluetooth.FLAG_NOTIFY | ubluetooth.FLAG_READ),#         ),#     ),# )# ((_srv_handle, _char_handle), ) = ble.gatts_register_services(services)NUS_UUID = 'AE25A5C1-4601-143C-12BB-8BC45A18749C'RX_UUID = 'AE25A5C2-4601-143C-12BB-8BC45A18749C'TX_UUID = 'AE25A5C3-4601-143C-12BB-8BC45A18749C'BLE_NUS = ubluetooth.UUID(NUS_UUID)BLE_RX = (ubluetooth.UUID(RX_UUID), ubluetooth.FLAG_WRITE)# BLE_TX = (ubluetooth.UUID(TX_UUID), ubluetooth.FLAG_NOTIFY)ELE_TX = (ubluetooth.UUID(TX_UUID), ubluetooth.FLAG_NOTIFY | ubluetooth.FLAG_READ)BLE_UART = (BLE_NUS, (BLE_TX, BLE_RX,))SERVICES = (BLE_UART, )((tx, rx),) = ble.gatts_register_services(SERVICES)# 获取特性的句柄以发送通知# char_handle = _char_handle[0]char_handle = tx# 启动广播ble.gap_advertise(100, bytearray(b'ESP32-C3 BLE Test'))return char_handle# BLE中断处理函数
def ble_irq(event, data):if event == ubluetooth.IRQ_GAP_CONNECTED:print("BLE device connected")elif event == ubluetooth.IRQ_GAP_DISCONNECTED:print("BLE device disconnected")# 可以选择在这里重新开始广播# 注册BLE中断处理函数
ble.irq(ble_irq)# 初始化BLE并获取特性句柄
char_handle = ble_init()# 发送通知的数据
notification_data = b'Hello, ESP32-C3 BLE!'
ble.gatts_notify(char_handle, notification_data)
print("Notification sent:", notification_data)# try:
#     while True:
#         # 检查是否有连接的设备
#         if ble.gatts_get_conn_handle():
#             # 发送通知
#             ble.gatts_notify(char_handle, notification_data)
#             print("Notification sent:", notification_data)#         # 等待一段时间再发送下一次通知
#         time.sleep(5)
# except KeyboardInterrupt:
#     print("Exiting")
# finally:
#     # 清理BLE资源
#     ble.active(False)
#     machine.deepsleep()  # 可选:使设备进入深度睡眠模式以节省电源

手机安装蓝牙测试app:BLE调试助手

 在华为手机的“应用市场”,查找“BLE调试助手”,安装该app应用。

打开“BLE调试助手”,找到开发板的蓝牙。小窍门,开发板的蓝牙,一般没有诸如“oppo”、小米、amazfit等字样,而且如果手机和开发板距离较近,信号会比较强,比如-50db左右

蓝牙通讯测试

在开发板发送数据,比如:

notification_data = b'Hello, ESP32-C3 BLE!'
ble.gatts_notify(char_handle, notification_data)notification_data = b'123456'
ble.gatts_notify(char_handle, notification_data)

这段代码可以让esp32c3发送信息,然后用手机app“BLE调试助手”接收,接收信息截图

最后六位数字,31-36,就是发送的数字1-6的ASCII编码。 

注意:不知道是否没有清空缓冲的原因,每点一次“接收”,都会再接收最后一次的信息。比如里面的连续多次31、32 ,就是测试时发送了“12”两个字母,但是点了多次接收导致的。

调试

这句有报错:>>> char_handle = ble_init()

Traceback (most recent call last): File "<stdin>", line 1, in <module> File "<stdin>", line 10, in ble_init ValueError: need more than 1 values to unpack

按照一言的提示,查看封包情况:

services = ((ubluetooth.UUID(SERVICE_UUID),((ubluetooth.UUID(CHARACTERISTIC_UUID), ubluetooth.FLAG_NOTIFY | ubluetooth.FLAG_READ),),),
)result = ble.gatts_register_services(services)
print(result)  # 打印结果以查看结构

输出:

((16,),)

感觉这个封包有问题啊,调整一下:

services = ((ubluetooth.UUID(SERVICE_UUID),((ubluetooth.UUID(CHARACTERISTIC_UUID), ubluetooth.FLAG_NOTIFY | ubluetooth.FLAG_READ),),),)((_srv_handle), ((_char_handle), _ ),), ) = ble.gatts_register_services(services)

封包和解包这里太头疼了,直接用下面的代码:

    NUS_UUID = 'AE25A5C1-4601-143C-12BB-8BC45A18749C'RX_UUID = 'AE25A5C2-4601-143C-12BB-8BC45A18749C'TX_UUID = 'AE25A5C3-4601-143C-12BB-8BC45A18749C'BLE_NUS = ubluetooth.UUID(NUS_UUID)BLE_RX = (ubluetooth.UUID(RX_UUID), ubluetooth.FLAG_WRITE)# BLE_TX = (ubluetooth.UUID(TX_UUID), ubluetooth.FLAG_NOTIFY)ELE_TX = (ubluetooth.UUID(TX_UUID), ubluetooth.FLAG_NOTIFY | ubluetooth.FLAG_READ)BLE_UART = (BLE_NUS, (BLE_TX, BLE_RX,))SERVICES = (BLE_UART, )((tx, rx),) = ble.gatts_register_services(SERVICES)# 获取特性的句柄以发送通知# char_handle = _char_handle[0]char_handle = tx

http://www.lryc.cn/news/487482.html

相关文章:

  • leetcode hot100【LeetCode 35.搜索插入位置】java实现
  • 我们要用平凡来诠释非凡
  • synchronized和volatile区别
  • 125.验证回文串-力扣(LeetCode)
  • 线程间通信:wait和notify
  • 风险识别和管理的工具
  • qt之QFTP对文件夹(含嵌套文件夹和文件)、文件删除下载功能
  • 为何数据库推荐将IPv4地址存储为32位整数而非字符串?
  • Mybatis框架之责任链模式 (Chain of Responsibility Pattern)
  • C++ Stack和Queue---单向守护与无尽等待:数据结构的诗意表达
  • 深入理解Java包装类与泛型的应用
  • 【机器学习chp4】特征工程
  • LeetCode螺旋矩阵
  • 第十五届蓝桥杯JAVA的B组题目详情解析
  • 在几分钟内将数据从 Oracle 迁移到 ClickHouse
  • ASP.NET MVC宠物商城系统
  • 完整http服务器
  • 【专题】2024AIGC创新应用洞察报告汇总PDF洞察(附原数据表)
  • 形态学图像处理(Morphological Image Processing)
  • 【IDER、PyCharm】免费AI编程工具完整教程:ChatGPT Free - Support Key call AI GPT-o1 Claude3.5
  • C++11的一些实用特性
  • 23种设计模式-观察者(Observer)设计模式
  • 【CUDA】Branch Divergence and Unrolling Loop
  • 深度学习:卷积神经网络的计算复杂度,顺序操作,最大路径长度
  • springboot 配置文件中 multipart.max-file-size 各个版本的写法
  • linux 中mysql查看慢日志
  • 单片机的基本组成与工作原理
  • 智慧隧道和智慧交通
  • List、Set、Map详解和区别
  • 界面控件DevExpress WinForms v24.2新功能预览 - 支持.NET 9