Arduino入门教程--第二十五课--EEPROM的读写与获取串口数据流
EEPROM (Electrically Erasable Programmable Read-Only Memory),电可擦可编程只读存储器,他是一种掉电后数据不丢失的存储芯片。具体能做什么呢?比如~数字密码保险箱的密码掉电后不能丢失,他一般都是存放在EEPROM里面的。再比如,一些设备需要根据不同场合,输入特定的环境参数,每次启动要加载这些参数,这些参数,也是放在EEPROM里面的。
下面的代码,我们是让Arduino不停地从0号地址位开始的读取EEPROM,把读取到的数值通过串口输出到电脑上。同时,我们通过串口向Arduino输入数字,更新从0号地址位开始的EEPROM信息。间接的更新了输出到电脑上的数值。
我们这次使用的是Arduino官方的EEPROM库,但是官方库有一个遗憾,每次只能读取或者写入一个地址位~~所以我们在代码开头。。定了了两个函数EEPROM_write与EEPROM_read。括号内第一个参数是起始地址位,第二个就是写入的变量。
#include <EEPROM.h>
#define EEPROM_write(address, p) {int i = 0; byte *pp = (byte*)&(p);for(; i < sizeof(p); i++) EEPROM.write(address+i, pp);}
#define EEPROM_read(address, p){int i = 0; byte *pp = (byte*)&(p);for(; i < sizeof(p); i++) pp=EEPROM.read(address+i);}
unsigned long incomingByte = 0; // 定义无符号长整数型变量incomingByte初始值为0
void setup() {
Serial.begin(9600); // 打开串口,设置数据传输速率9600
}
void loop() {
if (Serial.available() > 0) {
incomingByte = Serial.parseInt();//读取串口传入的下一个有效整数,把该整数赋值给incomingByte变量
EEPROM_write(0,incomingByte) //把incomingByte变量从0地址位开始写入EEPROM
}
EEPROM_read(0,incomingByte) //从0地址位开始读取EEPROM,把值写入变量incomingByte
Serial.println(incomingByte); //通过串口输出
delay(1000);
}
上效果视频~请点击全屏看。。。
http://player.youku.com/player.php/sid/XNTU5NjExNTky/v.swf
这次我们用到了一个串口命令Serial.parseInt(),他的作用是查找传入的串行数据流中的下一个有效的整数。Serial.read()只能每次读取一个字节。。。使用起来并不是很方便。Serial.parseInt()能一次读取一个数据流。。弥补了这方面使用上不便的缺陷。
Serial.parseInt()更多资料请查看WIKI chaoser 发表于 2013-8-24 23:33 static/image/common/back.gif
看到 byte *pp=(byte*)&p 的时候晕了一下,果然C语言的指针概念都还给老师了。
这里的意思应该是:定义了 ...
我来回答下==
1。我们下载的程序在 Flash空间里,不在EEPROM, 二者都可以断电保存数据,但是EEPROM的擦写次数(100000次)比Flash(10000次多)。。EEPROM可以在程序随时擦写,但是Flash不行(flash保存程序)。
2。最大字节表示可用程序存储空间,当然是Flash;
3。应该是可以放在括号里的;
4。注意 #define EEPROM_read(address,p)的关键字是define,所以这是个宏定义,而不是函数,所以不用加分号; 看到 byte *pp=(byte*)&p 的时候晕了一下,果然C语言的指针概念都还给老师了。
这里的意思应该是:定义了一个字节型数组指针pp,将它指向了变量p。(byte*)是为了将&p强制转换为字节型指针,以便赋值给*pp的。
整个程序是先从串口读了一个不知道哪儿来的数值,然后就把这个数值写入EEPROM,之后又从EEPROM里读出一个数,最后把这个数显示出来,让你看看跟输入数是否相同。
问题:
1、EEPROM初始存了些什么?是不是就是我们下载过去的程序?
2、IDE总是显示:“二进制程序大小:2,906字节(最大32,256字节)”,后面的最大字节数是否就是EEPROM的大小?
3、LZ定义的两个函数,int i=0; 完全可以放进for函数里,否则老以为其他什么地方还用了i变量。
4、在loop里调用这两个自定义函数时,结尾部分LZ都没加分号,居然还调试过了,这是怎么一个情况??
#include <EEPROM.h>
#define EEPROM_read(address,p) {byte *pp=(byte*)&p; for(int i=0; i<sizeof(p); i++) pp=EEPROM.read(address+i);}
#define EEPROM_write(address,p) {byte *pp=(byte*)&p; for(int i=0; i<sizeof(p); i++) EEPROM.write(address+i, pp);}
unsigned long incomingByte = 0; // 定义无符号长整数型变量incomingByte初始值为0
void setup() {Serial.begin(9600);}
void loop() {
if (Serial.available()>0) {
incomingByte=Serial.parseInt();//读取串口传入的下一个有效整数
EEPROM_write(0,incomingByte); //把incomingByte变量从0地址位开始写入EEPROM
}
EEPROM_read(0,incomingByte); //从0地址位开始读取EEPROM,把值写入变量incomingByte
Serial.println(incomingByte); //通过串口输出
delay(1000);
}
请我的板子怎么写入值后怎么只能读取一次呢??后面就变成了0.怎么回事? fairsky 发表于 2013-9-6 17:57 static/image/common/back.gif
请我的板子怎么写入值后怎么只能读取一次呢??后面就变成了0.怎么回事?
换用1.0.4或者1.0.1的IDE试试。。。1.0.5的IDE出现好多代码编译后运行异常问题。。。估计官方在里面改了很多东西 chaoser 发表于 2013-8-24 23:33 static/image/common/back.gif
看到 byte *pp=(byte*)&p 的时候晕了一下,果然C语言的指针概念都还给老师了。
这里的意思应该是:定义了 ...
顶贴!顶贴!顶贴! fairsky 发表于 2013-9-6 17:57 static/image/common/back.gif
请我的板子怎么写入值后怎么只能读取一次呢??后面就变成了0.怎么回事?
感觉是Serial.parseInt()留有未读完的数据(可能是串口工具的问题),下次串口有效,又读了一次,读出是零
在后面再加一条“Serial.parseInt();”,就没有这个问题了 可以对EEPROM进行空间划分分别存储一个数组吗???怎么实现啊??? 怎么对其是否写入过数据进行判断??? 强哥,编辑后出现这个,是什么情况呢。
sketch_nov27a.cpp: In function 'void loop()':
sketch_nov27a:14: error: 'class HardwareSerial' has no member named 'parseInt'
mark 下,以配以后使用 本帖最后由 Fortware 于 2014-4-14 16:00 编辑
弘毅大哥好 ,问个问题 宏定义里面
{int i = 0; byte *pp = (byte*)&(p);for(; i < sizeof(p); i++) EEPROM.write(address+i, pp);}
先获取把变量的地址,转化为byte类型的指针,然后复制给指针 *pp 引用时为什么不是
*pp (或者*(pp+i))而是 pp //抱歉,pp后面中括号显示不出来 本帖最后由 b9ss 于 2014-6-30 11:27 编辑
现学现用 想着这个不智能的IDE应该无法判断溢出,就试了试,结果发现EEPROM的实际寻址方式是用最大有效值取模。
//ATmega328
#include <EEPROM.h>
void setup() {
Serial.begin(9600);
EEPROM.write(0,110);
EEPROM.write(1023,119);
EEPROM.write(1024,911);
}
void loop() {
Serial.print("0::");
Serial.println(EEPROM.read(0));
Serial.print("1023::");
Serial.println(EEPROM.read(1023));
Serial.print("1024::");
Serial.println(EEPROM.read(1024));
}
/*输出
0::143
1023::119
1024::143
*/
b9ss 发表于 2014-6-30 11:22 static/image/common/back.gif
现学现用 想着这个不智能的IDE应该无法判断溢出,就试了试,结果发现EEPROM的实际寻址方式是用最大有效值取 ...
b9ss能帮我解释“EEPROM的实际寻址方式是用最大有效值取”这句话吗?还有输出结果我想不明白呢 迈步xxzj 发表于 2014-7-20 23:03 static/image/common/back.gif
b9ss能帮我解释“EEPROM的实际寻址方式是用最大有效值取”这句话吗?还有输出结果我想不明白呢
其实就是转圈圈 比如说一个人原地空翻了720度 其实最后姿态和原始姿态的角度差为0