在向 EEPROM 连续写入多个字节的数据时,如果每写一个字节都要等待几 ms 的话,整体上的写入效率就太低了。因此 EEPROM 的厂商就想了一个办法,把 EEPROM 分页管理。24C01、24C02 这两个型号是 8 个字节一个页,而 24C04、24C08、24C16 是 16 个字节一页。我们开发板上用的型号是 24C02,一共是 256 个字节,8 个字节一页,那么就一共有 32 页。 
  分配好页之后,如果我们在同一个页内连续写入几个字节后,最后再发送停止位的时序。EEPROM 检测到这个停止位后,就会一次性把这一页的数据写到非易失区域,就不需要像上节课那样写一个字节检测一次了,并且页写入的时间也不会超过 5ms。如果我们写入的数据跨页了,那么写完了一页之后,我们要发送一个停止位,然后等待并且检测 EEPROM 的空闲模式,一直等到把上一页数据完全写到非易失区域后,再进行下一页的写入,这样就可以在很大程度上提高数据的写入效率。 
  /*****************************I2C.c 文件程序源代码*******************************/
  (此处省略,可参考之前章节的代码)
  /***************************Lcd1602.c 文件程序源代码*****************************/
  (此处省略,可参考之前章节的代码) 
 
-   
- #include <reg52.h>  
-   
- extern void I2CStart();  
- extern void I2CStop();  
- extern unsigned char I2CReadACK();  
- extern unsigned char I2CReadNAK();  
- extern bit I2CWrite(unsigned char dat);  
-   
-   
- void E2Read(unsigned char *buf, unsigned char addr, unsigned char len){  
- do {   
-     I2CStart();  
-     if (I2CWrite(0x50<<1)){   
-         break;  
-     }  
-     I2CStop();  
- }while(1);  
-      
-     I2CWrite(addr);   
-     I2CStart();  
-     I2CWrite((0x50<<1)|0x01);   
-     while (len > 1){  
-         *buf++ = I2CReadACK();   
-         len--;  
-     }  
-     *buf = I2CReadNAK();   
-     I2CStop();  
- }  
-   
- void E2Write(unsigned char *buf, unsigned char addr, unsigned char len){  
-     while (len > 0){   
-         do {   
-             I2CStart();  
-             if (I2CWrite(0x50<<1)){   
-                 break;  
-             }  
-             I2CStop();  
-         } while(1);  
-       
-      
-         I2CWrite(addr);   
-         while (len > 0){  
-             I2CWrite(*buf++);   
-             len--;   
-             addr++;   
-               
-               
-             if ((addr&0x07) == 0){  
-                 break;   
-             }  
-         }  
-         I2CStop();  
-     }  
- }   
 
  遵循模块化的原则,我们把 EEPROM 的读写函数也单独写成一个 eeprom.c 文件。其中E2Read 函数和上一节是一样的,因为读操作与分页无关。重点是 E2Write 函数,我们在写入数据的时候,要计算下一个要写的数据的地址是否是一个页的起始地址,如果是的话,则必须跳出循环,等待 EEPROM 把当前这一页写入到非易失区域后,再进行后续页的写入。
 
-   
- #include <reg52.h>  
-   
- extern void InitLcd1602();  
- extern void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str);  
- extern void E2Read(unsigned char *buf, unsigned char addr, unsigned char len);  
- extern void E2Write(unsigned char *buf, unsigned char addr, unsigned char len);  
- void MemToStr(unsigned char *str, unsigned char *src, unsigned char len);  
-   
- void main(){  
-     unsigned char i;  
-     unsigned char buf[5];  
-     unsigned char str[20];  
-      
-     InitLcd1602();   
-     E2Read(buf, 0x8E, sizeof(buf));   
-     MemToStr(str, buf, sizeof(buf));   
-     LcdShowStr(0, 0, str);   
-     for (i=0; i<sizeof(buf); i++){   
-         buf[i] = buf[i] + 1 + i;  
-     }  
-     E2Write(buf, 0x8E, sizeof(buf));   
-     while(1);  
- }  
-  
-   
- void MemToStr(unsigned char *str, unsigned char *src, unsigned char len){  
-     unsigned char tmp;  
-     while (len--){  
-         tmp = *src >> 4;   
-         if (tmp <= 9){   
-             *str++ = tmp + '0';  
-         }else{  
-             *str++ = tmp - 10 + 'A';  
-         }  
-         tmp = *src & 0x0F;   
-         if (tmp <= 9){    
-             *str++ = tmp + '0';  
-         }else{  
-             *str++ = tmp - 10 + 'A';  
-         }  
-         *str++ = ' ';   
-         src++;  
-     }  
- }   
 
  多字节写入和页写入程序都编写出来了,而且页写入的程序我们还特地跨页写的数据,它们的写入时间到底差别多大呢。我们用一些工具可以测量一下,比如示波器,逻辑分析仪等工具。我现在把两次写入时间用逻辑分析仪给抓了出来,并且用时间标签 T1 和 T2 标注了开始位置和结束位置,如图 14-5 和图 14-6 所示,右侧显示的|T1-T2|就是最终写入 5 个字节所耗费的时间。多字节一个一个写入,每次写入后都需要再次通信检测 EEPROM 是否在“忙”,因此耗费了大量的时间,同样的写入 5 个字节的数据,一个一个写入用了 8.4ms 左右的时间,而使用页写入,只用了 3.5ms 左右的时间。                    