STC8增强型单片机进阶开发--EEPROM读写

2024-06-05 01:36

本文主要是介绍STC8增强型单片机进阶开发--EEPROM读写,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

知不足而奋进 望远山而前行


目录

系列文章目录

文章目录

前言

设置EEPROM

读写String字符串

官方示例

总结



前言

EEPROM是一种可擦写可编程只读存储器(Electrically Erasable Programmable Read-Only Memory)的缩写。它是一种非易失性存储器,可以在不需要外部电源的情况下保持存储数据。与ROM不同,EEPROM可以通过电子擦除和编程来修改存储的数据,因此它是一种可重写的存储器。


设置EEPROM

STC8H8K64U的EEPROM可以在烧录的时候指定大小, 如下图

读写String字符串

#include "Config.h"
#include "UART.h"
#include "EEPROM.h"
#include <string.h>void UART_config(void) {// >>> 记得添加 NVIC.c, UART.c, UART_Isr.c <<<COMx_InitDefine		COMx_InitStructure;					//结构定义COMx_InitStructure.UART_Mode      = UART_8bit_BRTx;	//模式, UART_ShiftRight,UART_8bit_BRTx,UART_9bit,UART_9bit_BRTxCOMx_InitStructure.UART_BRT_Use   = BRT_Timer1;			//选择波特率发生器, BRT_Timer1, BRT_Timer2 (注意: 串口2固定使用BRT_Timer2)COMx_InitStructure.UART_BaudRate  = 115200ul;			//波特率, 一般 110 ~ 115200COMx_InitStructure.UART_RxEnable  = ENABLE;				//接收允许,   ENABLE或DISABLECOMx_InitStructure.BaudRateDouble = DISABLE;			//波特率加倍, ENABLE或DISABLEUART_Configuration(UART1, &COMx_InitStructure);		//初始化串口1 UART1,UART2,UART3,UART4NVIC_UART1_Init(ENABLE,Priority_1);		//中断使能, ENABLE/DISABLE; 优先级(低到高) Priority_0,Priority_1,Priority_2,Priority_3UART1_SW(UART1_SW_P30_P31);		// 引脚选择, UART1_SW_P30_P31,UART1_SW_P36_P37,UART1_SW_P16_P17,UART1_SW_P43_P44
}#define     Max_Length          100      //读写EEPROM缓冲长度
u8  xdata   tmp[Max_Length];        //EEPROM操作缓冲void main() {u16 addr_sector = 0x0000;char *str = "HelloWorld!abc123!";u16 str_length = strlen(str);	// 获取str的长度UART_config();EA = 1;// 擦除扇区, 一次性擦除一个扇区512字节, 从0x0000开始, 0x01FF
//    EEPROM_SectorErase(u16 EE_address);EEPROM_SectorErase(addr_sector);//    // 写入数据. 字符串\int\long\floatEEPROM_write_n(u16 EE_address,u8 *DataAddress,u16 number);EEPROM_write_n(addr_sector, str, str_length);// 读取数据. 字符串\int\long\float
//    EEPROM_read_n(u16 EE_address,u8 *DataAddress,u16 number);EEPROM_read_n(addr_sector, tmp, str_length);// 添加字符串结束符tmp[str_length] = '\0';printf(">>存储的字符串: %s\n", str);printf(">>读到的字符串: %s\n", tmp);if(strcmp(str, tmp) == 0){printf("两个字符串相等\n");}else {printf("两个字符串不等\n");}while(1) {}}

官方示例

串口命令设置: (命令字母不区分大小写)

  • E 0x0040 --> 对0x0040地址扇区内容进行擦除.
  • W 0x0040 1234567890 --> 对0x0040地址写入字符1234567890.
  • R 0x0040 10 --> 对0x0040地址读出10个字节数据.
/*---------------------------------------------------------------------*/
/* --- STC MCU Limited ------------------------------------------------*/
/* --- STC 1T Series MCU Demo Programme -------------------------------*/
/* --- Mobile: (86)13922805190 ----------------------------------------*/
/* --- Fax: 86-0513-55012956,55012947,55012969 ------------------------*/
/* --- Tel: 86-0513-55012928,55012929,55012966 ------------------------*/
/* --- Web: www.STCAI.com ---------------------------------------------*/
/* --- BBS: www.STCAIMCU.com  -----------------------------------------*/
/* --- QQ:  800003751 -------------------------------------------------*/
/* 如果要在程序中使用此代码,请在程序中注明使用了STC的资料及程序            */
/*---------------------------------------------------------------------*/#include	"config.h"
#include	"STC8G_H_GPIO.h"
#include	"STC8G_H_UART.h"
#include	"STC8G_H_Delay.h"
#include	"STC8G_H_NVIC.h"
#include	"STC8G_H_EEPROM.h"
#include	"STC8G_H_Switch.h"/*************	本程序功能说明	**************本例程基于STC8H8K64U为主控芯片的实验箱8进行编写测试,STC8G、STC8H系列芯片可通用参考.通过串口对STC内部自带的EEPROM(FLASH)进行读写测试。对FLASH做扇区擦除、写入、读出的操作,命令指定地址。默认波特率:  115200,N,8,1. 串口命令设置: (命令字母不区分大小写)E 0x0040             --> 对0x0040地址扇区内容进行擦除.W 0x0040 1234567890  --> 对0x0040地址写入字符1234567890.R 0x0040 10          --> 对0x0040地址读出10个字节数据. 注意:下载时,下载界面"硬件选项"中设置用户EEPROM大小,并确保串口命令中的地址在EEPROM设置的大小范围之内。下载时, 选择时钟 22.1184MHz (可以在配置文件"config.h"中修改).******************************************/#define     Max_Length          100      //读写EEPROM缓冲长度/*************	本地常量声明	**************//*************	本地变量声明	**************/
u8  xdata   tmp[Max_Length];        //EEPROM操作缓冲/*************	本地函数声明	**************//*************  外部函数和变量声明 *****************//******************* IO配置函数 *******************/
void	GPIO_config(void)
{GPIO_InitTypeDef	GPIO_InitStructure;		//结构定义GPIO_InitStructure.Pin  = GPIO_Pin_0 | GPIO_Pin_1;		//指定要初始化的IO, GPIO_Pin_0 ~ GPIO_Pin_7GPIO_InitStructure.Mode = GPIO_PullUp;	//指定IO的输入或输出方式,GPIO_PullUp,GPIO_HighZ,GPIO_OUT_OD,GPIO_OUT_PPGPIO_Inilize(GPIO_P3,&GPIO_InitStructure);	//初始化
}/***************  串口初始化函数 *****************/
void	UART_config(void)
{COMx_InitDefine		COMx_InitStructure;					//结构定义COMx_InitStructure.UART_Mode      = UART_8bit_BRTx;	//模式, UART_ShiftRight,UART_8bit_BRTx,UART_9bit,UART_9bit_BRTxCOMx_InitStructure.UART_BRT_Use   = BRT_Timer1;			//选择波特率发生器, BRT_Timer1, BRT_Timer2 (注意: 串口2固定使用BRT_Timer2)COMx_InitStructure.UART_BaudRate  = 115200ul;			//波特率, 一般 110 ~ 115200COMx_InitStructure.UART_RxEnable  = ENABLE;				//接收允许,   ENABLE或DISABLECOMx_InitStructure.BaudRateDouble = DISABLE;			//波特率加倍, ENABLE或DISABLEUART_Configuration(UART1, &COMx_InitStructure);		//初始化串口1 UART1,UART2,UART3,UART4NVIC_UART1_Init(ENABLE,Priority_1);		//中断使能, ENABLE/DISABLE; 优先级(低到高) Priority_0,Priority_1,Priority_2,Priority_3
}/**********************************************/u8	CheckData(u8 dat)
{if((dat >= '0') && (dat <= '9'))		return (dat-'0');if((dat >= 'A') && (dat <= 'F'))		return (dat-'A'+10);if((dat >= 'a') && (dat <= 'f'))		return (dat-'a'+10);return 0xff;
}//========================================================================
// 函数: u16    GetAddress(void)
// 描述: 计算各种输入方式的地址.
// 参数: 无.
// 返回: 16位EEPROM地址.
// 版本: V1.0, 2013-6-6
//========================================================================
u16 GetAddress(void)
{u16 address;u8  i,j;address = 0;if((RX1_Buffer[2] == '0') && (RX1_Buffer[3] == 'X')){for(i=4; i<8; i++){j = CheckData(RX1_Buffer[i]);if(j >= 0x10)   return 65535;   //erroraddress = (address << 4) + j;}return (address);}return  65535;  //error
}/**************** 获取要读出数据的字节数 ****************************/
u8 GetDataLength(void)
{u8  i;u8  length;length = 0;for(i=9; i<COM1.RX_Cnt; i++){if(CheckData(RX1_Buffer[i]) >= 10)  break;length = length * 10 + CheckData(RX1_Buffer[i]);}return (length);
}/********************* 主函数 *************************/
void main(void)
{u8  i,j;u16 addr;u8  status;EAXSFR();		/* 扩展寄存器访问使能 */GPIO_config();UART_config();EA = 1;PrintString1("STC8系列单片机EEPROM测试程序,串口命令设置如下:\r\n");    //UART1发送一个字符串PrintString1("E 0x0040             --> 对0x0040地址扇区内容进行擦除\xfd.\r\n");     //UART1发送一个字符串PrintString1("W 0x0040 1234567890  --> 对0x0040地址写入字符1234567890.\r\n");  //UART1发送一个字符串PrintString1("R 0x0040 10          --> 对0x0040地址读出10个字节内容.\r\n");    //UART1发送一个字符串while(1){delay_ms(1);if(COM1.RX_TimeOut > 0)		//超时计数{if(--COM1.RX_TimeOut == 0){for(i=0; i<COM1.RX_Cnt; i++)    TX1_write2buff(RX1_Buffer[i]);    //把收到的数据原样返回,用于测试status = 0xff;  //状态给一个非0值if((COM1.RX_Cnt >= 8) && (RX1_Buffer[1] == ' ')) //最短命令为8个字节{for(i=0; i<8; i++){if((RX1_Buffer[i] >= 'a') && (RX1_Buffer[i] <= 'z'))    RX1_Buffer[i] = RX1_Buffer[i] - 'a' + 'A';  //小写转大写}addr = GetAddress();if(addr < 63488)    //限制在0~123扇区{if(RX1_Buffer[0] == 'E')    //写入N个字节{EEPROM_SectorErase(addr);           //擦除扇区PrintString1("擦除\xfd成功!\r\n");status = 0; //命令正确}else if((RX1_Buffer[0] == 'W') && (RX1_Buffer[8] == ' '))    //写入N个字节{j = COM1.RX_Cnt - 9;if(j > Max_Length)  j = Max_Length; //越界检测//EEPROM_SectorErase(addr);           //擦除扇区EEPROM_write_n(addr,&RX1_Buffer[9],j);      //写N个字节PrintString1("已写入");if(j >= 100)    {TX1_write2buff(j/100+'0');   j = j % 100;}if(j >= 10)     {TX1_write2buff(j/10+'0');    j = j % 10;}TX1_write2buff(j%10+'0');PrintString1("字节!\r\n");status = 0; //命令正确}else if((RX1_Buffer[0] == 'R') && (RX1_Buffer[8] == ' '))   //PC请求返回N字节EEPROM数据{j = GetDataLength();if(j > Max_Length)  j = Max_Length; //越界检测if(j > 0){PrintString1("读出");TX1_write2buff(j/10+'0');TX1_write2buff(j%10+'0');PrintString1("个字节内容如下:\r\n");EEPROM_read_n(addr,tmp,j);for(i=0; i<j; i++)  TX1_write2buff(tmp[i]);TX1_write2buff(0x0d);TX1_write2buff(0x0a);status = 0; //命令正确}}}}if(status != 0) PrintString1("命令错误!\r\n");COM1.RX_Cnt = 0;}}}
} 
/**********************************************/

总结

EEPROM通常用于存储需要频繁修改的数据,例如系统配置信息、用户设置、校准数据等。由于EEPROM可以在系统运行时进行读写操作,因此它在许多应用中都具有很高的实用价值。

这篇关于STC8增强型单片机进阶开发--EEPROM读写的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1031706

相关文章

SpringBoot实现数据库读写分离的3种方法小结

《SpringBoot实现数据库读写分离的3种方法小结》为了提高系统的读写性能和可用性,读写分离是一种经典的数据库架构模式,在SpringBoot应用中,有多种方式可以实现数据库读写分离,本文将介绍三... 目录一、数据库读写分离概述二、方案一:基于AbstractRoutingDataSource实现动态

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

Python基于wxPython和FFmpeg开发一个视频标签工具

《Python基于wxPython和FFmpeg开发一个视频标签工具》在当今数字媒体时代,视频内容的管理和标记变得越来越重要,无论是研究人员需要对实验视频进行时间点标记,还是个人用户希望对家庭视频进行... 目录引言1. 应用概述2. 技术栈分析2.1 核心库和模块2.2 wxpython作为GUI选择的优

利用Python开发Markdown表格结构转换为Excel工具

《利用Python开发Markdown表格结构转换为Excel工具》在数据管理和文档编写过程中,我们经常使用Markdown来记录表格数据,但它没有Excel使用方便,所以本文将使用Python编写一... 目录1.完整代码2. 项目概述3. 代码解析3.1 依赖库3.2 GUI 设计3.3 解析 Mark

利用Go语言开发文件操作工具轻松处理所有文件

《利用Go语言开发文件操作工具轻松处理所有文件》在后端开发中,文件操作是一个非常常见但又容易出错的场景,本文小编要向大家介绍一个强大的Go语言文件操作工具库,它能帮你轻松处理各种文件操作场景... 目录为什么需要这个工具?核心功能详解1. 文件/目录存javascript在性检查2. 批量创建目录3. 文件

Redis中高并发读写性能的深度解析与优化

《Redis中高并发读写性能的深度解析与优化》Redis作为一款高性能的内存数据库,广泛应用于缓存、消息队列、实时统计等场景,本文将深入探讨Redis的读写并发能力,感兴趣的小伙伴可以了解下... 目录引言一、Redis 并发能力概述1.1 Redis 的读写性能1.2 影响 Redis 并发能力的因素二、

基于Python开发批量提取Excel图片的小工具

《基于Python开发批量提取Excel图片的小工具》这篇文章主要为大家详细介绍了如何使用Python中的openpyxl库开发一个小工具,可以实现批量提取Excel图片,有需要的小伙伴可以参考一下... 目前有一个需求,就是批量读取当前目录下所有文件夹里的Excel文件,去获取出Excel文件中的图片,并

Java进阶学习之如何开启远程调式

《Java进阶学习之如何开启远程调式》Java开发中的远程调试是一项至关重要的技能,特别是在处理生产环境的问题或者协作开发时,:本文主要介绍Java进阶学习之如何开启远程调式的相关资料,需要的朋友... 目录概述Java远程调试的开启与底层原理开启Java远程调试底层原理JVM参数总结&nbsMbKKXJx

基于Python开发PDF转PNG的可视化工具

《基于Python开发PDF转PNG的可视化工具》在数字文档处理领域,PDF到图像格式的转换是常见需求,本文介绍如何利用Python的PyMuPDF库和Tkinter框架开发一个带图形界面的PDF转P... 目录一、引言二、功能特性三、技术架构1. 技术栈组成2. 系统架构javascript设计3.效果图

基于Python开发PDF转Doc格式小程序

《基于Python开发PDF转Doc格式小程序》这篇文章主要为大家详细介绍了如何基于Python开发PDF转Doc格式小程序,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 用python实现PDF转Doc格式小程序以下是一个使用Python实现PDF转DOC格式的GUI程序,采用T