|
發布時間: 2019-10-30 23:15
正文摘要:向epprom寫入數值2,讀取之后數碼管顯示,再把數值加1,重復寫入讀取,顯示這個步驟,一直加到99,為什么數碼管卻一直顯示剛開始的數值 2,求各位前輩指點! 下面把代碼附上: ———————————————— ... |
無良的魚 發表于 2019-10-31 12:05 官網查了下,起始地址是0x2000 ,記錯了 |
無良的魚 發表于 2019-10-31 12:05 你的第一個數字2寫進去了,所以IIC協議程序應該沒問題,所以問題就出在eeprom的擦除上,幫你百度了下,發現擦除還挺麻煩的,定義sfr地址啥的,沒記錯的話單片機的EEPROM起始地址應該是0x02000,也就是說0x02000可以寫進一個unsigned int型的數字,只要你的這個2定義的是uint型,那你完全可以每循環一次,寫入地址加1,然后讀取的地址也加1,即循環里addr++,而不是寫死成2,當然,這個0x02000這個扇區地址是有上限的,上限多少忘了,自己查。這個扇區只要不超過地址范圍就可以一直加。然后要注意的是,如果不執行擦除這一操作的話,數據應該是寫不進去的,eeprom是掉電可存儲,斷電數據不會丟失,如果這一地址里有數據,那么你的操作無效,值應該是不會變的,所以最好還是研究一下怎么擦除扇區,百度一下就出來了。 |
無良的魚 發表于 2019-10-31 12:05 好吧,習慣用16進制表示地址了,乍一看這個10進制的不太舒服,給你百度了下哈,以下是原話: 與單片機有關。 比如STC的單片機,在向片內EEPROM寫入數據前必須執行擦除操作,并且擦除是按照扇區進行的,并不能逐字節進行。 而STM8單片機,在向片內EEPROM寫入數據時不需要用戶在程序中進行擦除,單片機會自動擦除對應字節的數據。 所以你這個每次for循環都是直接寫入時肯定有問題的,IIC這個協議我用過很多次了都沒有問題,EEPROM的操作也有兩次,都沒有問題哈,你去網上重新搜一個IIC的協議程序再試試,記得寫入前要先擦除哈,這個真的很重要。 |
讀寫時序有問題吧,太亂了,看得人發暈,自己先梳理一下吧![]() |
無敵的UUZ大人 發表于 2019-10-31 14:13 但是好像就是這個地方出問題了 |
| I2C頂多是延時問題,直接示波器看波形就知道通信有沒錯了,如果有回復信號就看邏輯錯誤了 |
|
我也是剛學,一起研究一下。 我沒用過eeprom,但我知道你這么寫應該沒錯: for(k=2;k<100;k++) { Write(k,k); delay(1); num=Read(k); if(Ackflag) P1=0; else P1=0xff; display(num/10,num%10); } |
|
額,2就是epprom里面的第二個地址,就是向第二個地址里面寫入k的值,然后從第二個地址讀出來,用數碼管顯示, 然而是怎樣擦出的呢,我看的例程里面沒有這個過程,不怎么懂,應該是怎樣的呢 這個就是做練習用的,不停重復此過程, |
bai_yun_123 發表于 2019-10-31 08:48 2就是epprom里面的第二個地址,就是往第二個地址里面寫入k的值,然后從第二個地址里面讀出來然后用數碼管顯示 這個就是練習用的,不過怎么擦除呢,難道不是直接寫入就可以了嘛? |
| 我去,咋復制了這么一大坨,我記得就復制了一個for循環啊 |
|
向epprom寫入數值2,讀取之后數碼管顯示,再把數值加1,重復寫入讀取,顯示這個步驟,一直加到99,為什么數碼管卻一直顯示剛開始的數值 2,求各位前輩指點! 下面把代碼附上: ———————————————————————————————————————————————————————— #include <reg52.h> #include <intrins.h> sbit SCL=P3^4; sbit SDA=P3^5; bit Ackflag; char k; char code duan[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90}; char code we[]={0xfe,0xfd}; void delay5us() { _nop_(); } void init() { SCL=1; delay5us(); SDA=1; delay5us(); } void I2c_start() { SCL=1; delay5us(); SDA=1; delay5us(); SDA=0; delay5us(); } void I2c_stop() { SCL=0; SDA=0; delay5us(); SCL=1; delay5us(); SDA=1; delay5us(); } bit Read_Ack() { SCL=0; SCL=1; delay5us(); if(SDA) { SCL=0; return 1; } else { SCL=0; return 0; } } void Send_Ack(bit i) { SCL=0; if(i) SDA=1; else SDA=0; SCL=1; delay5us(); SCL=0; SDA=1; } void I2c_sendbyte(char dat) { char i; for(i=0;i<8;i++) { SCL=0; if(dat&0x80) SDA=1; else SDA=0; SCL=1; dat=dat<<1; } SCL=0; SDA=1; } char I2c_readbyte() { char i,dat; for(i=0;i<8;i++) { dat<<=1; SCL=0; SCL=1; if(SDA) dat|=0x01; } return(dat); } void Write(char addr,char dat) { I2c_start(); I2c_sendbyte(0xa0);//send a byte address if(Read_Ack()) Ackflag=1; else Ackflag=0; I2c_sendbyte(addr); if(Read_Ack()) Ackflag=1; else Ackflag=0; I2c_sendbyte(dat); if(Read_Ack()) Ackflag=1; else Ackflag=0; I2c_stop(); } char Read(char addr) { char dat; I2c_start(); I2c_sendbyte(0xa0);//send a byte address if(Read_Ack()) Ackflag=1; else Ackflag=0; I2c_sendbyte(addr); Read_Ack(); I2c_start(); I2c_sendbyte(0xa1); //read 1 if(Read_Ack()) Ackflag=1; else Ackflag=0; dat=I2c_readbyte(); Send_Ack(1); I2c_stop(); return (dat); } void delay(char z) { char x,y; for(x=z;x>0;x--) for(y=110;y>0;y--); } void display(char shi,char ge) { while(1) { P2=0xfd; P0=duan[ge]; delay(5); P2=0xfe; P0=duan[shi]; delay(5); } } void main() { char num; init(); TMOD=0x01; EA=1; ET0=1; TH0=(65536-46080)/256; TL0=(65536-46080)%256; TR0=1; for(k=2;k<100;k++) { Write(2,k); delay(1); num=Read(2); if(Ackflag) P1=0; else P1=0xff; display(num/10,num%10); } } 你這個write的第一個參數和read的參數都是address,這個是eeprom的扇區地址,不知道你寫個2是啥意思,然后eeprom每次寫入都需要擦除當前扇區里的東西,且每次操作的對象是當前整個扇區(一次擦除,整個扇區全部擦除,每個單片機的扇區數不一樣,具體百度資料),看你的for循環里也沒有在寫入之前先擦除,而且一般也不會每顯示一次都往扇區里寫入值,這樣多了個過程,當然練習就當我沒說,咋寫都成。 |