ESP8266/ESP32 NVS 基本操作

NVS 介绍
NVS: Non-volatile storage , 即将数据存储到 flash 中, 掉电或重启后数据仍然存在, flash 类似于 PC 上磁盘.
ESP8266 和 ESP32 上提供 nvs 接口给用户, 来保存和读取用户数据. 具体参考 nvs.h

nvs 更详细的说明请参考: NVS 文档

NVS 优势
接口更加安全
相比较于 spi_flash_read 和 spi_flash_write 等接口, NVS 不直接操作 address. 对于终端用户而已, 更加安全.
例如: 应用复杂一点, 容易 spi_flash_write(address, src, size) 不小心写到同一个地址, 或地址写覆盖, 而导致长时间 debug

接口使用接近用户习惯
NVS 接口类似于电脑上操作文件一样:
打开文件(nvs_open), 写文件(nvs_set_xxx), 保存文件(nvs_commit), 关闭文件(nvs_close)
打开文件(nvs_open), 读取文件(nvs_get_xxx), 关闭文件(nvs_close)

擦写均衡, 使 flash 寿命更长
NVS 在操作少量数据上, NVS 分区更大时, 擦写均衡表现的更为明显.

例如: flash 一个 sector 为 4KB, NVS 分配大小为一个 sector, 写同一个 64 Bytes 数据到 flash, 分别比较 spi_flash_xxx 和 nvs 写 64 次

spi_flash_write: 每次写 flash 前, 需擦除 flash. 对应: 64 次擦除 flash, 64 次写 flash
nvs: nvs 内部有擦写均衡, 有标志位记录当前有效存储. 如第一次擦除 sector, 再写 sector 0-63 Byte, 第二次写 sector 64-127 Bytes, 第 64 次(4KB/64Bytes) 写完 sector 最后一个 64 Byte. 对应: 1 次擦除 flash, 64 次写 flash
这样 NVS 减少 64 倍擦除操作, 对 flash 寿命有较大提升.
在 NVS 分区更大, 存储信息少时, 表现的更为明显.
NVS 写
更多参考: nvs.h

void nvs_write_data_to_flash(void)
{
    nvs_handle handle;
    static const char *NVS_CUSTOMER = "customer data";
    static const char *DATA1 = "param 1";
    static const char *DATA2 = "param 2";
    static const char *DATA3 = "param 3";

    int32_t value_for_store = 666;

     wifi_config_t wifi_config_to_store = {
        .sta = {
            .ssid = "store_ssid:hello_kitty",
            .password = "store_password:1234567890",
        },
    };

    printf("set size:%u\r\n", sizeof(wifi_config_to_store));
    ESP_ERROR_CHECK( nvs_open( NVS_CUSTOMER, NVS_READWRITE, &handle) );
    ESP_ERROR_CHECK( nvs_set_str( handle, DATA1, "i am a string.") );
    ESP_ERROR_CHECK( nvs_set_i32( handle, DATA2, value_for_store) );
    ESP_ERROR_CHECK( nvs_set_blob( handle, DATA3, &wifi_config_to_store, sizeof(wifi_config_to_store)) );

    ESP_ERROR_CHECK( nvs_commit(handle) );
    nvs_close(handle);

}

NVS 读
更多参考: nvs.h

void nvs_read_data_from_flash(void)
{
    nvs_handle handle;
    static const char *NVS_CUSTOMER = "customer data";
    static const char *DATA1 = "param 1";
    static const char *DATA2 = "param 2";
    static const char *DATA3 = "param 3";

    uint32_t str_length = 32;
    char str_data[32] = {0};
    int32_t value = 0;
    wifi_config_t wifi_config_stored;
    memset(&wifi_config_stored, 0x0, sizeof(wifi_config_stored));
    uint32_t len = sizeof(wifi_config_stored);

    ESP_ERROR_CHECK( nvs_open(NVS_CUSTOMER, NVS_READWRITE, &handle) );

    ESP_ERROR_CHECK ( nvs_get_str(handle, DATA1, str_data, &str_length) );
    ESP_ERROR_CHECK ( nvs_get_i32(handle, DATA2, &value) );
    ESP_ERROR_CHECK ( nvs_get_blob(handle, DATA3, &wifi_config_stored, &len) );

    printf("[data1]: %s len:%u\r\n", str_data, str_length);
    printf("[data2]: %d\r\n", value);
    printf("[data3]: ssid:%s passwd:%s\r\n", wifi_config_stored.sta.ssid, wifi_config_stored.sta.password);

    nvs_close(handle);
}

其他说明
不限于上面功能, nvs.h 中提供更多接口给用户使用
可参考 esp-idf nvs blob demo 来测试相关接口和使用
NVS 说明文档
--------------------- 
作者:ustccw 
来源:CSDN 
原文:https://blog.csdn.net/ustccw/article/details/84636394 
版权声明:本文为博主原创文章,转载请附上博文链接!

ESP32的Non-Volatile Storage (NVS)是一种存储系统,可以在ESP32中存储和检索数据,即使在断电重启后也可以保留数据。备份ESP32的NVS数据非常重要,以防止意外丢失数据。 以下是备份ESP32 NVS数据的步骤: 1. 使用ESP-IDF提供的nvs_flash_init函数初始化NVS系统。 2. 使用nvs_open函数打开NVS命名空间。 3. 使用nvs_get_blob函数获取要备份的数据。要备份所有数据,可以使用nvs_get_all函数。 4. 关闭NVS命名空间并调用nvs_flash_deinit函数。 5. 将备份数据写入外部存储器,如SD卡。可以使用ESP-IDF提供的SD卡驱动程序来实现。 下面是一个简单的示例代码,演示如何备份ESP32的NVS数据到SD卡上: ```c #include "esp_system.h" #include "nvs_flash.h" #include "nvs.h" #include "driver/sdmmc_host.h" #include "sdmmc_cmd.h" #include "esp_vfs_fat.h" #define NVS_NAMESPACE "my_namespace" #define SD_MOUNT_POINT "/sdcard" void backup_nvs_data() { // 初始化NVS系统 esp_err_t ret = nvs_flash_init(); if (ret == ESP_ERR_NVS_NO_FREE_PAGES || ret == ESP_ERR_NVS_NEW_VERSION_FOUND) { ESP_ERROR_CHECK(nvs_flash_erase()); ret = nvs_flash_init(); } ESP_ERROR_CHECK(ret); // 打开NVS命名空间 nvs_handle_t nvs_handle; ret = nvs_open(NVS_NAMESPACE, NVS_READONLY, &nvs_handle); ESP_ERROR_CHECK(ret); // 获取要备份的数据 size_t nvs_size = 0; ret = nvs_get_blob(nvs_handle, "my_data", NULL, &nvs_size); ESP_ERROR_CHECK(ret); uint8_t* nvs_data = malloc(nvs_size); ret = nvs_get_blob(nvs_handle, "my_data", nvs_data, &nvs_size); ESP_ERROR_CHECK(ret); // 关闭NVS命名空间 nvs_close(nvs_handle); // 卸载SD卡 esp_vfs_fat_sdmmc_unmount(); // 挂载SD卡 sdmmc_host_t host = SDMMC_HOST_DEFAULT(); sdmmc_slot_config_t slot_config = SDMMC_SLOT_CONFIG_DEFAULT(); esp_vfs_fat_sdmmc_mount_config_t mount_config = { .format_if_mount_failed = true, .max_files = 5 }; sdmmc_card_t* card; esp_err_t card_ret = esp_vfs_fat_sdmmc_mount(SD_MOUNT_POINT, &host, &slot_config, &mount_config, &card); ESP_ERROR_CHECK(card_ret); // 将备份数据写入SD卡 FILE* fp = fopen("/sdcard/nvs_backup.bin", "wb"); fwrite(nvs_data, nvs_size, 1, fp); fclose(fp); // 卸载SD卡 esp_vfs_fat_sdmmc_unmount(); // 释放内存 free(nvs_data); } ``` 在这个例子中,我们使用了ESP-IDF提供的SD卡驱动程序来挂载和卸载SD卡,并使用标准的文件IO函数来写入备份数据。 请注意,此代码仅用于演示目的。在实际应用中,您需要根据您的需求进行修改和调整。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值