13、ESP32 深度睡眠

1、深度睡眠

        ESP32 可以在不同的电源模式之间切换:

  • 活动模式
  • 调制解调器睡眠模式
  • 浅睡眠模式
  • 深度睡眠模式
  • 休眠模式

         在深度睡眠模式下,CPU 或 Wi-Fi 活动都不会发生,但超低功耗 (ULP) 协处理器仍可打开电源,RTC 内存也会保持通电状态,因此我们可以为 ULP 编写程序并将其存储在 RTC 内存中,以访问外围设备、内部定时器和内部传感器。

        如果需要通过外部事件、计时器唤醒主 CPU,同时保持最小的功耗,则此操作模式非常有用。

        在深度睡眠期间,ULP 协处理器可以使用部分 ESP32 引脚,即 RTC_GPIO 引脚和 Touch 引脚。

        深度睡眠设置步骤:

  1. 首先需要配置唤醒源,可以使用一个或组合多个唤醒源。
  2. 可以决定在深度睡眠期间关闭或保持打开哪些外围设备。默认情况下,ESP32 会自动关闭定义的唤醒源不需要的外设。
  3. 最后,使用 esp_deep_sleep_start() 将 ESP32 置于深度睡眠模式的功能。

2、唤醒源

        将 ESP32 置于深度睡眠模式后,有几种方法可以唤醒它:

  • 使用计时器,使用预定义的时间段唤醒 ESP32;
  • 使用触摸引脚;
  • 使用外部唤醒:可以使用一种外部唤醒,也可以使用几种不同的外部唤醒;
  • 使用 ULP 协处理器唤醒。

       

        进入睡眠函数:

esp_deep_sleep_start();

         返回唤醒原因函数:

esp_sleep_get_wakeup_cause();


3、定时器唤醒

        ESP32 RTC 控制器具有内置定时器,可用于在预定义的时间后唤醒 ESP32。

        ESP32 可以进入深度睡眠模式,然后在预定义的时间段唤醒。如果正在运行需要时间戳或日常任务的项目,同时保持低功耗,则此功能特别有用。
        启用定时器唤醒只需在以下函数中指定休眠时间(单位微秒):esp_sleep_enable_timer_wakeup(time_in_us);

#include <Arduino.h>


// 使用定时器唤醒时,通电的部分是 RTC 控制器、RTC 外设和 RTC 存储器。

#define uS_TO_S_FACTOR 1000000  /* 微秒到秒的转换系数 */
#define TIME_TO_SLEEP  5        /* ESP32休眠时间(单位秒) */

RTC_DATA_ATTR int bootCount = 0;    // RTC 内存上定义深度睡眠中唤醒的次数 bootCount


// 唤醒原因
void print_wakeup_reason()
{
    esp_sleep_wakeup_cause_t wakeup_reason;

    wakeup_reason = esp_sleep_get_wakeup_cause();

    switch(wakeup_reason)
    {
        case ESP_SLEEP_WAKEUP_EXT0 : Serial.println("Wakeup caused by external signal using RTC_IO"); break;
        case ESP_SLEEP_WAKEUP_EXT1 : Serial.println("Wakeup caused by external signal using RTC_CNTL"); break;
        case ESP_SLEEP_WAKEUP_TIMER : Serial.println("Wakeup caused by timer"); break;
        case ESP_SLEEP_WAKEUP_TOUCHPAD : Serial.println("Wakeup caused by touchpad"); break;
        case ESP_SLEEP_WAKEUP_ULP : Serial.println("Wakeup caused by ULP program"); break;
        default : Serial.printf("Wakeup was not caused by deep sleep: %d\n",wakeup_reason); break;
    }
}


void setup()
{
    Serial.begin(115200);
    delay(2000);        // 打开串行监视器

    
    ++bootCount;    // 每次重新启动时计数
    Serial.println("Boot number: " + String(bootCount));

    print_wakeup_reason();      // 打印ESP32的唤醒原因

    esp_sleep_enable_timer_wakeup(TIME_TO_SLEEP * uS_TO_S_FACTOR);      // 配置唤醒源(微秒单位)
    Serial.println("Setup ESP32 to sleep for every " + String(TIME_TO_SLEEP) +" Seconds");


    Serial.println("Going to sleep now");
    delay(1000);
    Serial.flush(); 
    esp_deep_sleep_start();     // 进入睡眠
    Serial.println("This will never be printed");
}

void loop()
{
}

4、触摸唤醒

        启用触摸唤醒只需调用函数:

esp_sleep_enable_touchpad_wakeup();

#include <Arduino.h>


// 触摸引脚设置阈值
#if CONFIG_IDF_TARGET_ESP32
    #define THRESHOLD     40        /* 值越大,灵敏度越高 */
#else       // ESP32-S2 and ESP32-S3 + default for other chips (to be adjusted) */
    #define THRESHOLD     5000      /* 值越小,灵敏度越高 */
#endif

RTC_DATA_ATTR int bootCount = 0;    // RTC 内存上定义深度睡眠中唤醒的次数 bootCount
touch_pad_t touchPin;


// 唤醒原因
void print_wakeup_reason()
{
    esp_sleep_wakeup_cause_t wakeup_reason;

    wakeup_reason = esp_sleep_get_wakeup_cause();

    switch(wakeup_reason)
    {
        case ESP_SLEEP_WAKEUP_EXT0 : Serial.println("Wakeup caused by external signal using RTC_IO"); break;
        case ESP_SLEEP_WAKEUP_EXT1 : Serial.println("Wakeup caused by external signal using RTC_CNTL"); break;
        case ESP_SLEEP_WAKEUP_TIMER : Serial.println("Wakeup caused by timer"); break;
        case ESP_SLEEP_WAKEUP_TOUCHPAD : Serial.println("Wakeup caused by touchpad"); break;
        case ESP_SLEEP_WAKEUP_ULP : Serial.println("Wakeup caused by ULP program"); break;
        default : Serial.printf("Wakeup was not caused by deep sleep: %d\n",wakeup_reason); break;
    }
}


// 被唤醒的触摸 IO
void print_wakeup_touchpad()
{
    touchPin = esp_sleep_get_touchpad_wakeup_status();

    #if CONFIG_IDF_TARGET_ESP32
    switch(touchPin)
    {
        case 0  : Serial.println("Touch detected on GPIO 4"); break;
        case 1  : Serial.println("Touch detected on GPIO 0"); break;
        case 2  : Serial.println("Touch detected on GPIO 2"); break;
        case 3  : Serial.println("Touch detected on GPIO 15"); break;
        case 4  : Serial.println("Touch detected on GPIO 13"); break;
        case 5  : Serial.println("Touch detected on GPIO 12"); break;
        case 6  : Serial.println("Touch detected on GPIO 14"); break;
        case 7  : Serial.println("Touch detected on GPIO 27"); break;
        case 8  : Serial.println("Touch detected on GPIO 33"); break;
        case 9  : Serial.println("Touch detected on GPIO 32"); break;
        default : Serial.println("Wakeup not by touchpad"); break;
    }
    #else
        if(touchPin < TOUCH_PAD_MAX)
        {
            Serial.printf("Touch detected on GPIO %d\n", touchPin); 
        }
        else
        {
            Serial.println("Wakeup not by touchpad");
        }
    #endif
}


void setup()
{
    Serial.begin(115200);

    ++bootCount;    // 每次重新启动时计数
    Serial.println("Boot number: " + String(bootCount));

    print_wakeup_reason();      // 打印唤醒原因
    print_wakeup_touchpad();    // 打印触摸 IO

    #if CONFIG_IDF_TARGET_ESP32 
        //Setup sleep wakeup on Touch Pad 3 + 7 (GPIO15 + GPIO 27) 
        touchSleepWakeUpEnable(T3,THRESHOLD);
        touchSleepWakeUpEnable(T7,THRESHOLD);
    #else //ESP32-S2 + ESP32-S3
        //Setup sleep wakeup on Touch Pad 3 (GPIO3) 
        touchSleepWakeUpEnable(T3,THRESHOLD);
    #endif

    Serial.println("Going to sleep now");
    esp_deep_sleep_start();     // 进入睡眠
    Serial.println("This will never be printed");
}

void loop()
{
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

常见的数据结构

链表 链表&#xff1a;适用于插入删除多、读少的场景。 链表在新增、删除数据都比较容易&#xff0c;可以在 O(1) 的时间复杂度内完成。 但对于查找&#xff0c;不管是按照位置的查找还是按照数值条件的查找&#xff0c;都需要对全部数据进行遍历。这显然就是 O(n) 的时间复杂…

解锁流量密码:如何利用HubSpot打造高效的获客策略?(下)

在当今数字化时代&#xff0c;流量是企业成功的关键。HubSpot作为一款全面的营销自动化工具&#xff0c;为我们提供了强大的支持&#xff0c;帮助企业打造高效的流量获取策略。接下来&#xff0c;我们将从社交媒体与SEO优化、自动化营销流程、数据分析与效果评估以及流量获取策…

畅通工程(并查集)

//新生训练 #include <iostream> #include <algorithm> #include <bits/stdc.h> using namespace std;#define N 1010int fa[N];int Find(int x) {if (x ! fa[x]){fa[x] Find(fa[x]);}return fa[x]; }void Union(int x, int y) {int a Find(x);int b Find…

Linux读写文件

前言 学习了文件系统&#xff0c;就能理解为什么说Linux下一切皆文件。 语言层面的操作 在c语言的学习中我们可以使用fopen()函数对文件进行操作。 int main() {//FILE * fp fopen("./log.txt", "w");//FILE * fp fopen("./log.txt", "…

Compose 布局

文章目录 Compose 布局ColumnColumn属性使用 RowRow属性使用 BoxBox属性使用 ConstraintLayoutLazyColumnLazyColumn属性使用使用多类型使用粘性标题回到顶部 LazyRowLazyRow属性使用 LazyVerticalGridLazyVerticalGrid属性使用 Compose 布局 Column Compose中的”垂直线性布…

BlueNRG-X 原理图参数说明

1. 前言 为了让客户在原理图设计阶段少走弯路&#xff0c;我这里结合客户评估和设计阶段常遇到的问题&#xff0c;整理了一下 BlueNRG-1/-2 相关设计及注意事项以备客户解惑用。 2. BlueNRG-1/-2 的原理图参数说明及设计注意事项 2.1. BlueNRG-1/-2 原理图及参数如下&#x…

NFTScan | 04.08~04.14 NFT 市场热点汇总

欢迎来到由 NFT 基础设施 NFTScan 出品的 NFT 生态热点事件每周汇总。 周期&#xff1a;2024.04.08~ 2024.04.14 NFT Hot News 01/ 数据&#xff1a;Runestone 地板价突破 0.07 BTC&#xff0c;创历史新高 4 月 8 日&#xff0c;据数据显示&#xff0c;Runestone 地板价突破 …

【第七届openGauss技术文章征集】 openGauss新版本征文活动来啦!

活动背景 2024年3月30日&#xff0c;openGauss 6.0.0版本正式上线&#xff0c;该版本与之前版本特性功能保持兼容&#xff0c;在内核能力、DataPod三层资源池化架构、DataKit数据全生命周期管理平台、生态兼容性等方面全面增强。&#xff08;下方【点击原文】即可查看更多【新…

Android--ConnectivityManager使用

一、前言 Android10之后官方废弃了通过WifiManager连接WIFI的方式&#xff0c;现在要使用ConnectivityManager连接WIFI 二、连接WIFI public class MainActivity extends AppCompatActivity {private static final String TAG"lkx";Overrideprotected void onCrea…

【黑马头条】-day10热点文章定时计算-xxl-job

文章目录 1 今日内容1.1 需求分析1.2 解决方案1.3 定时计算1.4 定时任务方案对比 2 分布式任务调度3 xxl-job3.1 简介3.2 环境搭建3.2.1 配置maven3.2.2 源码说明 3.3 配置部署调度中心3.3.1 运行sql脚本3.3.2 修改配置application.properties3.3.3 启动引导类 3.4 docker配置x…

✌粤嵌—2024/4/18—旋转链表✌

代码实现&#xff1a; 方法一&#xff1a;在原链表中找到旋转之后的首尾&#xff0c;改变指向 /*** Definition for singly-linked list.* struct ListNode {* int val;* struct ListNode *next;* };*/ struct ListNode* rotateRight(struct ListNode *head, int k) {i…

电脑怎么创建加密磁盘?方法很简单!

在电脑中创建加密磁盘&#xff0c;可以有效地保护电脑数据&#xff0c;避免数据泄露。那么&#xff0c;电脑怎么创建加密磁盘呢&#xff1f;下面我们就一起来了解一下吧。 密盘创建方法 创建密盘建议使用超级秘密磁盘3000&#xff0c;软件界面简约&#xff0c;操作简单&#x…

秋招复习笔记——八股文部分:网络HTTP

常见面试题 基本概念 HTTP 是超文本传输协议&#xff0c;也就是HyperText Transfer Protocol。HTTP 协议是一个双向协议,是一个在计算机世界里专门用来在两点之间传输数据的约定和规范。是超越了普通文本的文本&#xff0c;它是文字、图片、视频等的混合体&#xff0c;最关键…

资源管理规范

踩坑经验&#xff1a; 对于IO及池化资源(文件、线程池、网络IO(HttpClient)、磁盘IO)&#xff0c;使用之后一定要及时回收&#xff0c;好借好还&#xff0c;再借不难。 稳定关闭方式 完成I/O操作后&#xff0c;应该关闭流以释放系统资源。可以使用finally块确保流被关闭&#…

学生故事|勇于创新,拒绝“一成不变”的设计

对于JIANG MANQI而言&#xff0c;室内设计一直是她钟爱的行业选择。 在没有进入莱佛士学习之前&#xff0c;MANQI受家人的影响&#xff0c;从小就对设计行业比较感兴趣。而她选择室内设计&#xff0c;是觉得室内设计是比较有前途的一个专业&#xff0c;随着人们生活品质的提高…

数字化转型-工具变量数据集

01、数据介绍 数字化转型是指企业或个人利用数字技术&#xff0c;如大数据、云计算、人工智能等&#xff0c;对其业务流程、运营模式、决策方式等进行全面、深入的变革&#xff0c;以提高效率、降低成本、提升质量、增强竞争力。在这个过程中&#xff0c;工具变量扮演着至关重…

SpringBoot 集成Nacos注册中心和配置中心-支持自动刷新配置

SpringBoot 集成Nacos注册中心和配置中心-支持自动刷新配置 本文介绍SpringBoot项目集成Nacos注册中心和配置中心的步骤&#xff0c;供各位参考使用 1、配置pom.xml 文件 在pom.xml文件中定义如下配置和引用依赖&#xff0c;如下所示&#xff1a; <properties><pr…

MATLAB求和函数

语法 S sum(A) S sum(A,“all”) S sum(A,dim) S sum(A,vecdim) S sum(,outtype) S sum(,nanflag) 说明 示例 S sum(A) 返回沿大小大于 1 的第一个数组维度计算的元素之和。 如果 A 是向量&#xff0c;则 sum(A) 返回元素之和。 如果 A 是矩阵&#xff0c;则 sum(A) 将…

如何在Linux CentOS部署宝塔面板并实现固定公网地址访问内网宝塔

文章目录 一、使用官网一键安装命令安装宝塔二、简单配置宝塔&#xff0c;内网穿透三、使用固定公网地址访问宝塔 宝塔面板作为建站运维工具&#xff0c;适合新手&#xff0c;简单好用。当我们在家里/公司搭建了宝塔&#xff0c;没有公网IP&#xff0c;但是想要在外也可以访问内…

Spring Boot 中如何处理存取 MySQL 中 JSON 类型的字段

&#x1f468;&#x1f3fb;‍&#x1f4bb; 热爱摄影的程序员 &#x1f468;&#x1f3fb;‍&#x1f3a8; 喜欢编码的设计师 &#x1f9d5;&#x1f3fb; 擅长设计的剪辑师 &#x1f9d1;&#x1f3fb;‍&#x1f3eb; 一位高冷无情的全栈工程师 欢迎分享 / 收藏 / 赞 / 在看…