基于esp32实现键值对存储读写c程序例程

在基于 ESP32 的系统中,我们可以使用 NVS(Non-Volatile Storage,非易失性存储)来实现系统配置参数的掉电存储和读写。NVS 是 ESP32 提供的一种存储机制,允许我们将键值对数据存储在闪存中,即使设备掉电,数据也不会丢失。
以下是一个简单的 C 程序例程,展示了如何使用 NVS 进行系统配置参数的读写:

#include <stdio.h>
#include "nvs_flash.h"
#include "nvs.h"

// 定义一个函数用于写入配置参数
esp_err_t write_config_parameter(const char* key, int value) {
    // 打开 NVS 命名空间
    nvs_handle_t my_handle;
    esp_err_t err = nvs_open("storage", NVS_READWRITE, &my_handle);
    if (err != ESP_OK) {
        printf("Error (%s) opening NVS handle!\n", esp_err_to_name(err));
        return err;
    }

    // 写入参数
    err = nvs_set_i32(my_handle, key, value);
    if (err != ESP_OK) {
        printf("Error (%s) writing to NVS!\n", esp_err_to_name(err));
    } else {
        // 提交更改
        err = nvs_commit(my_handle);
        if (err != ESP_OK) {
            printf("Error (%s) committing to NVS!\n", esp_err_to_name(err));
        }
    }

    // 关闭 NVS 句柄
    nvs_close(my_handle);
    return err;
}

// 定义一个函数用于读取配置参数
esp_err_t read_config_parameter(const char* key, int* value) {
    // 打开 NVS 命名空间
    nvs_handle_t my_handle;
    esp_err_t err = nvs_open("storage", NVS_READONLY, &my_handle);
    if (err != ESP_OK) {
        printf("Error (%s) opening NVS handle!\n", esp_err_to_name(err));
        return err;
    }

    // 读取参数
    err = nvs_get_i32(my_handle, key, value);
    switch (err) {
        case ESP_OK:
            printf("Read parameter %s: %d\n", key, *value);
            break;
        case ESP_ERR_NVS_NOT_FOUND:
            printf("Parameter %s not found!\n", key);
            break;
        default :
            printf("Error (%s) reading from NVS!\n", esp_err_to_name(err));
    }

    // 关闭 NVS 句柄
    nvs_close(my_handle);
    return err;
}

void app_main() {
    // 初始化 NVS
    esp_err_t err = nvs_flash_init();
    if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) {
        // NVS 分区被截断,需要擦除并重试
        ESP_ERROR_CHECK(nvs_flash_erase());
        err = nvs_flash_init();
    }
    ESP_ERROR_CHECK(err);

    // 定义要读写的键和值
    const char* key = "config_param";
    int write_value = 123;

    // 写入配置参数
    err = write_config_parameter(key, write_value);
    if (err == ESP_OK) {
        printf("Parameter %s written successfully!\n", key);
    }

    // 读取配置参数
    int read_value;
    err = read_config_parameter(key, &read_value);
    if (err == ESP_OK) {
        printf("Parameter %s read successfully: %d\n", key, read_value);
    }
}


代码说明:
1. 初始化 NVS:在 app_main 函数中,我们首先调用 nvs_flash_init() 来初始化 NVS。如果 NVS 分区被截断或有新版本,我们需要擦除并重试。
2. 写入配置参数:write_config_parameter 函数用于将一个整数参数写入 NVS。我们首先打开 NVS 命名空间,然后使用 nvs_set_i32 写入参数,最后调用 nvs_commit 提交更改。
3. 读取配置参数:read_config_parameter 函数用于从 NVS 中读取一个整数参数。我们打开 NVS 命名空间,使用 nvs_get_i32 读取参数,并根据返回值进行相应的处理。
4. 主函数:在 app_main 函数中,我们调用 write_config_parameter 写入一个参数,然后调用 read_config_parameter 读取该参数,并打印结果。
注意事项:
• 这个例程只是一个简单的示例,实际应用中可能需要处理更多的数据类型和错误情况。
• NVS 有一定的存储空间限制,需要合理使用。
• 在写入和读取参数时,要确保键的唯一性。

=====================大容量数据存储方案==============================

在ESP32中备份OTA升级的bin文件,下面为你分析不同存储方案以及是否可以使用NVS(Non-Volatile Storage):

NVS(非易失性存储)

不适合的原因
  • 容量限制:NVS主要用于存储少量的键值对数据,比如设备的配置信息、校准参数等。ESP32的NVS分区通常容量较小(一般只有几十KB),而OTA升级的bin文件通常较大(可能有几百KB甚至几MB),NVS无法满足存储bin文件的容量需求。
  • 设计目的:NVS是为了方便存储和管理小型的、频繁读写的配置数据而设计的,并非用于存储大文件。它的读写操作是基于键值对的,不适合处理大尺寸的二进制数据。

其他可行的存储方案及对应库

内置闪存(使用SPIFFS或LittleFS)
  • 特点:ESP32自带一定容量的内置闪存(常见4MB、8MB等),无需额外硬件,读写速度较快,但容量有限。
  • 对应库及使用说明
    • SPIFFS(SPI Flash File System):早期常用于ESP32的文件系统。使用时,需要在代码里进行初始化,之后就能像操作普通文件系统一样读写文件。
    #include "esp_spiffs.h"
    #include <stdio.h>
    
    void init_spiffs() {
        esp_vfs_spiffs_conf_t conf = {
          .base_path = "/spiffs",
          .partition_label = NULL,
          .max_files = 5,
          .format_if_mount_failed = true
        };
        esp_err_t ret = esp_vfs_spiffs_register(&conf);
        if (ret != ESP_OK) {
            if (ret == ESP_FAIL) {
                printf("Failed to mount or format filesystem\n");
            } else if (ret == ESP_ERR_NOT_FOUND) {
                printf("Failed to find SPIFFS partition\n");
            } else {
                printf("Failed to initialize SPIFFS (%s)\n", esp_err_to_name(ret));
            }
        }
    }
    
    void backup_bin_file(const uint8_t* bin_data, size_t data_size) {
        FILE* f = fopen("/spiffs/ota_backup.bin", "w");
        if (f == NULL) {
            printf("Failed to open file for writing\n");
            return;
        }
        fwrite(bin_data, 1, data_size, f);
        fclose(f);
    }
    
    • LittleFS:是一种较新的文件系统,相比SPIFFS有更好的性能和稳定性。使用方式与SPIFFS类似,先初始化,再进行文件操作。
    #include "esp_littlefs.h"
    #include <stdio.h>
    
    esp_err_t init_littlefs() {
        esp_vfs_littlefs_conf_t conf = {
          .base_path = "/littlefs",
          .partition_label = NULL,
          .format_if_mount_failed = true,
          .dont_mount = false
        };
        esp_err_t err = esp_vfs_littlefs_register(&conf);
        if (err != ESP_OK) {
            if (err == ESP_FAIL) {
                printf("Failed to mount or format filesystem\n");
            } else if (err == ESP_ERR_NOT_FOUND) {
                printf("Failed to find LittleFS partition\n");
            } else {
                printf("Failed to initialize LittleFS (%s)\n", esp_err_to_name(err));
            }
            return err;
        }
        return ESP_OK;
    }
    
    void backup_bin_to_littlefs(const uint8_t* bin_data, size_t data_size) {
        if (init_littlefs() == ESP_OK) {
            FILE* f = fopen("/littlefs/ota_backup.bin", "w");
            if (f == NULL) {
                printf("Failed to open file for writing\n");
                return;
            }
            fwrite(bin_data, 1, data_size, f);
            fclose(f);
        }
    }
    
外部SD卡
  • 特点:有较大存储容量,可备份多个OTA升级bin文件,成本较低,但读写速度相对内置闪存可能较慢,且需要额外引脚用于SPI通信。
  • 对应库及使用说明:使用ESP-IDF的SD卡驱动库,要先进行SD卡初始化,然后进行文件操作。
#include "driver/sdmmc_host.h"
#include "sdmmc_cmd.h"
#include <stdio.h>

void backup_to_sd(const uint8_t* bin_data, size_t data_size) {
    sdmmc_host_t host = SDMMC_HOST_DEFAULT();
    sdmmc_slot_config_t slot_config = SDMMC_SLOT_CONFIG_DEFAULT();

    esp_err_t err = esp_vfs_fat_sdmmc_mount("/sdcard", &host, &slot_config, NULL);
    if (err != ESP_OK) {
        if (err == ESP_FAIL) {
            printf("Failed to mount filesystem. If you want the card to be formatted, set format_if_mount_failed = true.\n");
        } else {
            printf("Failed to initialize the card (%s). Make sure SD card lines have pull - up resistors in place.\n", esp_err_to_name(err));
        }
        return;
    }

    FILE* f = fopen("/sdcard/ota_backup.bin", "w");
    if (f == NULL) {
        printf("Failed to open file for writing\n");
    } else {
        fwrite(bin_data, 1, data_size, f);
        fclose(f);
    }
}
外部SPI Flash
  • 特点:通过SPI接口连接外部SPI Flash芯片可扩展存储容量,读写速度较快,体积小,但需要额外的SPI Flash芯片和相应驱动程序。
  • 对应库及使用说明:使用ESP-IDF的SPI Flash驱动库,依据具体的SPI Flash芯片型号进行配置和操作,不同芯片操作方式有差异,具体代码需参考芯片数据手册。
<think>嗯,用户想了解ESP32的分区表,特别是.csv和.bin格式的区别,以及几个子分区的功能。首先,我需要回忆一下ESP-IDF的分区表相关知识。记得分区表是用来定义Flash存储布局的,不同的分区存放不同的数据,比如应用程序、NVS、OTA更新等。 首先,用户提到.csv和.bin格式的区别。.csv应该是文本文件,用来配置分区的名称、类型、子类型、偏移量和大小等信息,而.bin是编译后生成的实际二进制分区表文件,供ESP32启动时读取。这个需要确认一下官方文档的说法,确保正确性。例如,ESP-IDF的文档中提到分区表通常是CSV格式,然后通过工具转换成二进制。 接下来是各个子分区的作用。用户列举了nvs、phy_init、factory、vfs、storage这些子分区。需要逐个分析: 1. **nvs分区**:非易失性存储存储键值对数据,比如Wi-Fi配置、设备参数等。可能还涉及加密的情况,比如引用[2]中提到的生成加密的nvs.bin文件。 2. **phy_init分区**:存储PHY初始化数据,即射频相关的校准参数,确保无线通信正常工作。 3. **factory分区**:出厂应用程序,通常是默认固件,OTA更新时可能作为回滚分区。 4. **vfs分区**:虚拟文件系统,可能用于挂载SPIFFS或LittleFS等文件系统,存储用户文件。 5. **storage分区**:通用存储,可能用户自定义数据,比如日志或配置。 需要确认每个分区的类型和子类型是否正确。例如,nvs的类型是data,子类型是nvs;factory的类型是app,子类型是factory。这些信息可能在引用[1]中有详细说明。 还要注意分区表中的偏移量和大小设置,是否有对齐要求,比如分区通常需要4KB对齐。此外,用户可能想知道如何创建和修改这些分区,是否需要特殊工具,比如gen_esp32part.py将CSV转换为BIN。 用户可能的问题可能涉及如何配置这些分区,或者在实际项目中如何应用,比如OTA更新时如何处理分区。需要确保回答准确,并引用相关参考资料中的信息,比如引用[1]和[2]提到的分区表概述和NVS键值对的配置。 最后,生成相关问题时,要考虑用户可能的后续疑问,比如分区表如何影响OTA,如何调整分区大小,或NVS分区的加密方法等。这些都需要在相关问题中体现,帮助用户进一步探索。</think>### ESP32分区表格式及子分区功能解析 #### 一、分区表格式:`.csv`与`.bin`的区别 1. **`.csv`格式** - 是文本文件,用于定义分区的名称、类型、偏移量、大小等参数。 - 示例: ``` # Name, Type, SubType, Offset, Size, Flags nvs, data, nvs, 0x9000, 0x4000, phy_init, data, phy, 0xd000, 0x1000, factory, app, factory, 0x10000, 1M, ``` - 特点:可读性强,支持注释,便于手动编辑[^1]。 2. **`.bin`格式** - 是二进制文件,由`.csv`文件通过工具(如`gen_esp32part.py`)生成。 - 作用:直接写入Flash,供ESP32启动时加载分区表。 - 特点:包含校验信息,确保分区表完整性[^1]。 #### 二、关键子分区功能说明 1. **`nvs`分区** - **类型**:`data`,子类型`nvs` - **功能**:存储键值对形式的非易失性数据(如Wi-Fi配置、设备参数)。 - **特点**:支持加密(如通过`nvs_partition_gen.py`生成加密的`.bin`文件)[^2]。 2. **`phy_init`分区** - **类型**:`data`,子类型`phy` - **功能**:存储射频(RF)校准数据,用于Wi-Fi/BLE物理层初始化。 - **特点**:通常由厂商工具生成,用户无需修改。 3. **`factory`分区** - **类型**:`app`,子类型`factory` - **功能**:存放出厂固件,系统默认从此分区启动。 - **特点**:OTA更新时作为回滚备份分区。 4. **`vfs`分区** - **类型**:`data`,子类型`fat`或`spiffs` - **功能**:挂载虚拟文件系统(如SPIFFS),存储用户文件(如网页资源、日志)。 - **示例配置**: ``` storage, data, spiffs, 0x200000, 0x100000 ``` 5. **`storage`分区** - **类型**:自定义`data`类型 - **功能**:通用存储区域,可存放用户自定义数据(如传感器记录、配置模板)。 - **灵活性**:需在代码中手动管理读写操作。 #### 三、分区表设计注意事项 - **偏移量对齐**:建议按4KB对齐(如`0x1000`的倍数)。 - **空间预留**:OTA场景需至少两个应用程序分区(如`ota_0`和`ota_1`)。 - **加密需求**:敏感数据分区(如`nvs`)可通过`espsecure.py`工具加密[^2]。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值