物联网开发笔记(53)- 使用Micropython开发ESP32开发板之蓝牙BLE通信

news2025/1/22 21:55:18

一、目的

        这一节我们学习如何使用我们的ESP32开发板通过蓝牙和手机进行通信。

二、环境

        ESP32 + 手机(笔者用的小米10) + Thonny IDE

三、蓝牙介绍

        这个知识大家自行百度吧,这里不再赘述什么是蓝牙和蓝牙的历史,以及相关的专业知识。

四、手机蓝牙调试APP

        用的最多的是这三个:nRF Connect、BLE调试助手、LightBlue

笔者用的是:BLE调试助手

为什么用这个?因为小米手机应用商店直接可以安装,其他两个没搜到,哈哈。。。大家按需安装哈!

nRF Connect GitHub 地址:

GitHub - nrfconnect/sdk-nrf: nRF Connect SDK main repositorynRF Connect SDK main repository. Contribute to nrfconnect/sdk-nrf development by creating an account on GitHub.https://github.com/nrfconnect/sdk-nrf

五、调试代码

        大家先把下面的代码,拷贝到Thonny IDE中,保存到ESP32开发板中。

from machine import Pin
from machine import Timer
from time import sleep_ms
import bluetooth

BLE_MSG = ""  # 定义一个空的变量


class ESP32_BLE():
    def __init__(self, name):
        self.led = Pin(2, Pin.OUT)  # 创建LED对象,此LED为板载的蓝色LED,它用的是GPIO口2
        self.timer1 = Timer(0)  # 定时器,ESP32有4个硬件定时器,此处使用的是0
        self.name = name  # 传进来的参数name
        self.ble = bluetooth.BLE()  # 创建BLE对象
        self.ble.active(True)  # 启动蓝牙
        self.ble.config(gap_name=name)  # 配置蓝牙,给蓝牙起个名字
        self.disconnected()  # 调用函数执行计时器
        self.ble.irq(self.ble_irq)  #蓝牙调用中断函数。当手机发送数据给ESP32, ESP32蓝牙收到数据后自动执行此中断
        self.register()  # 注册
        self.advertiser()  # 广播

    def connected(self):
        self.led.value(1)   # 值为1表示蓝色LED亮
        self.timer1.deinit()  # 取消计时器

    def disconnected(self):
        # 初始化定时器,设定周期100ms,模式为周期性的,回调函数
        self.timer1.init(period=100, mode=Timer.PERIODIC, callback=lambda t: self.led.value(not self.led.value()))

    def ble_irq(self, event, data):
        global BLE_MSG
        if event == 1: #_IRQ_CENTRAL_CONNECT 手机已连接此设备
            self.connected()  # 调用函数,实现蓝牙没连上时闪烁,连上后长亮的效果
        elif event == 2: #_IRQ_CENTRAL_DISCONNECT 手机已断开此设备
            self.advertiser()  # 调用函数,进行蓝牙广播
            self.disconnected()  # 蓝牙断开连接后,蓝色LED又会闪烁
        elif event == 3: #_IRQ_GATTS_WRITE 手机发送数据给此设备 
            buffer = self.ble.gatts_read(self.rx)  # 接收来自手机的数据
            BLE_MSG = buffer.decode('UTF-8').strip()  # 将受到的数据安装utf-8进行解码,所以手机发送数据时应该为utf-8编码
            
    def register(self):        
        service_uuid = '6E400001-B5A3-F393-E0A9-E50E24DCCA9E'
        reader_uuid = '6E400002-B5A3-F393-E0A9-E50E24DCCA9E'
        sender_uuid = '6E400003-B5A3-F393-E0A9-E50E24DCCA9E'
        
        # 蓝牙是通过服务来工作的
        services = (
            # 定义一个控制灯的服务
            (   # 这个括号里的代表一个服务
                bluetooth.UUID(service_uuid),  # 服务的ID
                (
                    (bluetooth.UUID(sender_uuid), bluetooth.FLAG_NOTIFY),  # 服务类型:通知
                    (bluetooth.UUID(reader_uuid), bluetooth.FLAG_WRITE),   # 服务类型:写入
                )
            ),
            # 下面也可以定义别的服务
            # -- 其他服务
        )

        ((self.tx, self.rx,), ) = self.ble.gatts_register_services(services)

    def send(self, data): 
        self.ble.gatts_notify(0, self.tx, data + '\n')  # 蓝牙发送通知信息

    def advertiser(self):  # 当断开连接时,开始广播
        name = bytes(self.name, 'UTF-8')  # 将设备名字转为utf-8格式
        adv_data = bytearray('\x02\x01\x02') + bytearray((len(name) + 1, 0x09)) + name
        self.ble.gap_advertise(100, adv_data)
        print(adv_data)
        print("\r\n")


def buttons_irq(pin):  # 创建BOOT按键函数
    led.value(not led.value())
    print('LED is ON.' if led.value() else 'LED is OFF')
    ble.send('LED is ON.' if led.value() else 'LED is OFF')


if __name__ == "__main__":
    ble = ESP32_BLE("ESP32 BLE")  # 创建一个对象,带1个实参,后面用来给BLE起名字用

    but = Pin(0, Pin.IN)  # 创建对象,ESP32开发板的USB接口的右边的BOOT按键的GPIO口为0
    but.irq(trigger=Pin.IRQ_FALLING, handler=buttons_irq)  # 设置中断

    led = Pin(2, Pin.OUT) # 创建蓝色LED对象

    while True:
        if BLE_MSG == 'led state':  # 如果ESP32收到手机发来的数据
            print("Query: " + BLE_MSG)  # 打印出读取到的信息
            BLE_MSG = ""
            print('LED is on\n' if led.value() else 'LED is off\n')  # 如果灯亮输出LED is on.
            sleep_ms(8000)  # 设置为5秒,大家发送查询命令后,手机赶紧切换到通知页面,等待接收ESP32发送的通知信息
            ble.send('LED state is on' if led.value() else 'LED state is off')  # 蓝牙发送led数据给手机,如果灯亮发给手机“LED is on.”
        elif BLE_MSG == 'led on':  # 如果ESP32收到手机发来的数据
            print("Command: " + BLE_MSG)  # 打印出读取到的信息
            BLE_MSG = "" 
            if led.value() != 1:
                led.value(1)
            print('LED state is on\n' if led.value() else 'LED state is off\n')
        elif BLE_MSG == 'led off':  # 如果ESP32收到手机发来的数据
            print("Command: " + BLE_MSG)  # 打印出读取到的信息
            BLE_MSG = ""
            if led.value() != 0:
                led.value(0)
            print('LED state is on\n' if led.value() else 'LED state is off\n')

        sleep_ms(100)

六、调试效果

        我们在Thonny中运行代码,会看到ESP32 的蓝色LED灯开始闪烁

 

 

 Thonny IDE 打印信息:

 然后我们在手机上打开BLE调试助手,连接我们的开发板

 然后看到显示“连接成功”

 

此时ESP32开发板的蓝色LED灯停止闪烁 ,变为长亮

 然后我们就可以读取一些信息,比如获得开发板的名字:

 然后我们就可以控制开发板的蓝色LED灯,查询灯的状态和控制灯的亮灭

 

然后我们切换到通知页面,代码会返回查询的结果:

 

关灯:

 开灯:

 

然后我们在Thonny IDE中看到如下打印信息:

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/40479.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

JS(第二十六)ES6语法中function

JS(第九课)深刻的去理解函数._星辰镜的博客-CSDN博客 1 Function函数的定义 方式1 函数声明方式 function 关键字 (命名函数) function fn(){} 方式2 函数表达式(匿名函数) var fn function(){} 方式3 new Function() var f new Function(a, b, console.log(a b)); f(1, …

Tomcat服务器和Web开发介绍

Tomcat服务器和Web开发介绍 一、开启Web开发 什么是web开发 WEB,即网页的意思,它用于表示Internet主机上供外界访问的资源。 Internet上供外界访问的Web资源分为: 静态web资源(如html 页面):指web页面中供…

数据挖掘期末复习

考点目录 文章目录考点目录复习准备1. 数据挖掘的标准流程2. 数据挖掘的主要功能3. 数据探索的主要内容及其意义数据质量分析1.异常值分析2.缺失值分析数据特征分析1.分布分析6.相关性分析4. 数据预处理的作用及其主要任务5. 常见的噪声处理方法6.常用的缺失值处理方法7. 常用的…

Map和Set的详解

Map和Set是一种专门用来搜素的容器或者数据结构,其搜索的效率与其具体的实例化子类有关,是一种适合动态查找的集合容器 一、模型 一般把搜索的数据称为关键字(Key),和关键字对应的称为值(Value)…

Teams Tab App 的 manifest 分析

上一篇文章我们深入来分析了 tab app 的代码,这篇文章我们研究一下 manifest。 Teams 的 manifest 实际上是一个很复杂的 json,里面的配置非常多,之前的文章陆陆续续的讲过一些配置,我们在这篇文章里来看看关于 tabs 的一些配置&…

链表之删除单链表中的重复节点

文章目录删除单链表中的重复节点题目描述解题思路代码实现删除单链表中的重复节点 力扣链接 题目描述 编写代码,移除未排序链表中的重复节点。保留最开始出现的节点。 示例1:输入:[1, 2, 3, 3, 2, 1]输出:[1, 2, 3]示例2:输入:…

WEB网站安全检测系统设计与实现

目 录 1 引言 1 2 Web服务器所受的威胁及防御 1 2.1 缓冲区溢出 1 2.2 SQL注入攻击 1 2.3 基于脚本的DDos攻击 2 2.4 其他的不安全因素 3 3 Web的木马检测系统的设计 4 3.1 体系结构 4 3.2 处理流程 5 3.3 对客户端访问的响应 7 3.4 策略引擎的设计 8 3.4.1 策略的属性 8 3.4.2…

9、软件包管理

文章目录9、软件包管理9.1 RPM9.1.1 RPM 概述9.1.2 RPM 查询命令(rpm -qa)9.1.3 RPM 卸载命令(rpm -e)9.1.4 RPM 安装命令(rpm -ivh)9.2 YUM 仓库配置9.2.1 YUM 概述9.2.2 YUM 的常用命令9.2.3 修改网络 YU…

Linux系统中基本的启动方式

大家好, 今天主要和大家聊一聊,Linux系统的启动方式有哪些? 目录 第一:启动方式基本简介 第二​:启动模式的选择 第一:启动方式基本简介 Linux系统支持多种启动方式,可以从SD/EMMC、NAND Flas…

Jumperserver堡垒机管理服务器实战

一、 Jumpserver堡垒机简介 1、跳板机简介 跳板机就是一台服务器,开发或运维人员在维护过程中首先要统一登录到这台服务器,然后再登录到目标设备进行维护和操作。 跳板机缺点:没有实现对运维人员操作行为的控制和审计,使用跳板机的过程中还是会出现误操作、违规操作导致的…

Vue实现角色权限动态路由详细教程,在vue-admin-template基础上修改,附免费完整项目代码

前言 vue-admin-template是一个最基础的后台管理模板,只包含了一个后台需要最基础的东西,如果clone的是它的master分支,是没有权限管理的,只有完整版vue-element-admin有这个功能,但是为了小小的一个权限管理而用比较…

java框架 Spring之 AOP 面向切面编程 切入点表达式 AOP通知类型 Spring事务

AOP(Aspect Oriented Programming)面向切面编程,一种编程范式,指导开发者如何组织程序结构 作用:在不惊动原始设计的基础上为其进行功能增强 Spring理念:无入侵式/无侵入式 我们在不修改源代码的时候,为了执行另外的…

idea如何快速找到项目中对应的类(包括源码)

文章目录1. 前言2. 先说结论3. idea的全局搜索功能 MethodValidation4. 搜索spring源码(例子)1. 前言 最近在看某些功能的时候,会去看对应的源码,而有时候只知道类名,不知道从哪里进入源码,因此就比较好奇,idea的全局…

JAVA类加载器

JAVA是一种解释型语言,也就是一种边解释边执行的语言。JAVA所有源代码在执行之前,先要被编译成class文件,然后类加载器加载解析class文件,最后才执行。 JVM自带了几个类型的类加载器,JVM使用分层的软件架构方式设计类…

蜂鸟E203学习笔记(二)--蜂鸟E203总体框架

蜂鸟E203总体框架 蜂鸟E203处理器系统如下图所示 一、蜂鸟E203处理器核设计总览和顶层 1.1 蜂鸟E203处理器核的设计理念 模块化和可重用性:将处理器分成几个主体模块,每个单元之间的接口简单清晰。面积最小化:追求低功耗和小面积&#x…

DS18B20

一、DS18B20初始化时序图 (1) 先将数据线置高电平“1”。 (2) 延时(该时间要求的不是很严格,但是尽可能的短一点)。 (3) 数据线拉到低电平“0”。 (4&#xf…

【配送路径规划】基于matlab遗传算法求解静态外卖骑手路径规划问题【含Matlab源码 2248期】

⛄一、遗传算法求解静态外卖骑手路径规划问题 1 模型假设 外卖配送的实际运行是一个复杂的过程, 受诸多因素影响, 为了建立调度模型, 本文做如下假设。 (1) 外卖配送更多的是服务特殊群体, 所以本文认为外卖配送是一种预约型配送, 即在进行调度安排前, 己经获取了所有顾客的地…

Java并发-生产者消费者实现

生产者与消费者模型介绍 定义: 生产者消费者模式是一个十分经典的多线程并发协作的模式。 意义:弄懂生产者消费者问题能够让我们对并发编程的理解加深。 介绍:所谓生产者 - 消费者问题,实际上主要是包含了两类线程,…

Python面向对象编程之对象行为与特殊方法

面向对象编程之对象行为与特殊方法 python中的对象通常根据它们的行为和实现的功能进行分类。例如,所有序列类型都分在一组,如字符串,列表和元组,就是因为它们都支持一组相同的序列操作,如s[n], len[s]等。 所有基本…

【学习总结】LSD-SLAM配置与运行记录

今天安装测试了LSD-SLAM,记录配置中遇到的问题。 LSD-SLAM论文 LSD-SLAM: Large-Scale Direct Monocular SLAM, J. Engel, T. Schps, D. Cremers, ECCV 14 Semi-Dense Visual Odometry for a Monocular Camera, J. Engel, J. Sturm, D. Cremers, ICCV 13 配置环境…