求知 文章 文库 Lib 视频 iPerson 课程 认证 咨询 工具 讲座 Modeler   Code  
会员   
要资料
 
追随技术信仰

随时听讲座
每天看新闻
 
 
单片机教程
第一章 单片机入门
1.单片机简介
2.开发板选择
3.前期准备
第二章 LED及入门
1.初识原理图
2.程序点灯
3.实际LED硬件连接
4.点亮LED
5.闪烁的LED
6.软件调试查看运行时间
7.延时1秒
8.函数封装
9.函数传参调用
10.流水灯
11.数组与移位
第三章 蜂鸣器
1.蜂鸣器原理
2.无源蜂鸣器鸣叫
3.无源蜂鸣器题目
4.无源蜂鸣器+LED
第四章 数码管
1.数码管原理
2.数组&数码管
3.数码管&LED
4.多个数码管显示
5.同时显示不同的数字
6.高位不显示0
7.数码管小数点
8.分钟秒表
第五章 独立按键
1.按键入门
2.按键&蜂鸣器&数码管
3.按键猜想
4.按键用法
5.按键模式
6.不支持连按
7.静态变量
8.支持连按与全局变量
9.再次优化不支持连按的代码
10.流水灯的优化解说
11. 代码对比
12.双模式函数封装
13.第一阶段综合例程(上)
14.第一阶段综合例程(下)
15.第一阶段的总结
第六章 多文件编程
1.多文件编写
2.模块初始化
3.数码管显示函数
4.多文件编程首次测试
5.带返回值的函数
6.新按键程序
7.最终按键程序
8.最终的function文件
第七章 外部中断
1.寄存器
2.中断函数
3.进一步理解中断函数
4.中断的实验现象
5.外部中断测脉冲个数
第八章 定时器
1.定时器概念
2.定时器工作模式
3.定时时长的做法
4.定时器简单运用
5.定时器工作模式2
6.定时器中断函数的使用
7.定时器初始化新写法
8.隐形漏洞
9.代码参考
10.微调定时精确时间
11.单独文件封装
12.输入捕获
13.数据类型强制转换
14.定时器&数码管扫描显示
15.呼吸灯
第九章 舵机与超声波模块
1.舵机入门
2.舵机与按键
3.超声波模块
第十章 串口通信
1.串口通信入门
2.简洁式串口通信
3.详细理解ASCII码
4.串口printf系列函数
第十一章 1602液晶屏
1.液晶屏代码讲解
2.指向数组的指针
3.液晶屏代码单独文件
4.各功能代码的运用
第十二章 IIC通信
1.IIC入门代码讲解
2.EEPROM简单使用
3.IIC&EEPROM合成文件
第十三章 红外遥控与温度传感器
1.红外遥控
2.代码文件和测试
3.温度传感器代码解析Ⅰ
4.温度传感器代码解析Ⅱ
5.温度传感器代码独立文件
第十四章 AD与DA
1.AD与DA合成文件
第十五章 混合例程
1.电位器控制舵机
2.遥控器控制舵机
3.温度传感器与串口
4.模拟倒车雷达报警
5.再次熟悉串口
6.串口&液晶屏Ⅰ
7.串口&液晶屏Ⅱ
8.串口&液晶屏Ⅲ
9.串口&EEPROM
10.实践操作
11.结束语
 
 
目录
串口&EEPROM
来源:C语言网    作者:继承叔    编辑:Alice(火龙果软件)
414 次浏览
3次  

1.题目要求

作为本教程最后一个例程,我们也不打算写的太复杂,这次还是运用串口的知识,实现用串口往EEPROM写入数据,同时也能用串口读出EEPROM里的数据出来显示在电脑端的窗口上。根据前几讲的串口与液晶屏的代码理解,这一讲的例程也不难理解。

2.main.c测试代码

#include <reg52.h> 
#include <function.h> //详见第六章第8讲
#include <uart.h>     //详见第十章第4讲
#include <iic.h>      //详见第十二章第3讲
  
u8 uart_i=0;
u8 pdata RXDBUF[130];
void COUNT_TIMES()
{
    TMOD&=0xF0;   //清0低四位
    TMOD|=0x01;   //设置定时器0为工作模式1
}
    
void UART_E2Write()
{
    u8 j,k;
    u8 E2_add;
    u8 pdata E2BUF[102];//存取从RXDBUF[12]开始后的数据,RXDBUF[0]~RXDBUF[11]是字符串“E2Write:xxx,”(xxx表示地址)
    if(  RXDBUF[0]=='E'&&RXDBUF[1]=='2'&&RXDBUF[2]=='W'&&RXDBUF[3]=='r'
       &&RXDBUF[4]=='i'&&RXDBUF[5]=='t'&&RXDBUF[6]=='e'&&RXDBUF[7]==':')                               
    {
        E2_add=(RXDBUF[8]-'0')*100+(RXDBUF[9]-'0')*10+(RXDBUF[10]-'0'); //确定要开始写入的地址  
        k=uart_i-12;      //由于uart_i在“InterruptUART()”函数中的最后自加了一次,
                          //所以“k=uart_i-12;”表示的是收到要写入EEPROM的字符串的个数
          
        for(j=0; j<k; j++)E2BUF[j]=RXDBUF[j+12];       //把接收到要写入EEPROM的字符串存储在E2BUF数组中
     
        E2Write(E2BUF,E2_add,k);            //把内容写入进EEPROM中
        printf_str("\r\n写入完成!\r\n");
        printf_str("地址写到了");
        printf_num( (u32)E2_add+j-1 );     //打印出此时EEPROM写到了哪个地址位置了,因为要传入的参数为u32类型,所以必须强制转换
        printf_rn();   
    }
}
   
void UART_E2Read()
{
    u8 E2_add,n;
    u8 pdata E2BUF[102];//接收缓冲区,最多能接收50个汉字
    if(  RXDBUF[0]=='E'&&RXDBUF[1]=='2'&&RXDBUF[2]=='R'&&RXDBUF[3]=='e'
       &&RXDBUF[4]=='a'&&RXDBUF[5]=='d'&&RXDBUF[6]==':')                               
    {
        E2_add=(RXDBUF[7]-'0')*100+(RXDBUF[8]-'0')*10+(RXDBUF[9]-'0'); //确定要开始读出的地址  
        n=(RXDBUF[11]-'0')*100+(RXDBUF[12]-'0')*10+(RXDBUF[13]-'0');   //确定要读出的个数
     
        E2Read(E2BUF,E2_add,n);        
        E2BUF[n]='\0';     //添加结束符
     
        printf_str(E2BUF); //打印出从EEPROM读出的数据    
    }
}
   
void main()
{   
    u8 x; 
    EA = 1;        //闭合总中断开关
    COUNT_TIMES(); //初始化计数器
    ConfigUART(9600);
    
    while(1)
    {    
        if(TH0>=200)   //判断是否把数据接收完毕
        {
            TR0=0;     //关闭定时器,停止计时
      
            UART_E2Write(); //查看串口发送来的信息是否满足EEPROM写入数据的命令
            UART_E2Read();  //查看串口发送来的信息是否满足EEPROM读出数据的命令
      
            uart_i=0;    //uart_i归0是为了下次接收新的字符串是从RXDBUF[0]开始存取数据
            TH0=0;       //清零定时器的寄存器,下次再从0开始计时  
            TL0=0;     
      
            for(x=0;x<64;x++)RXDBUF[x]=0x00; //发送完接收到的字符串之后,内存区全部清除数据,统一改为0x00以便下次接收新的字符串
        }
    
}
    
void InterruptUART() interrupt 4
{  
    if (RI)  //接收到字节
    {
        RI = 0;
      
        TH0=0;//每接收到一个字节数据就清0寄存器的值,定时时间又是从0开始计时,直到TH0大于等于200的时候,
        TL0=0;//时间超过50ms,也就是没有再接收到数据,因为接收到数据都会进入“if (RI)”把TH0和TL0清0,TH0大于等于200,
              //超出了等待时间,证明后面没有数据发送过来了    
        TR0=1;//接收到第一个数据就打开定时器开始计时,后面接收到第二,第三个数据照样打开定时器计时,直到在主函数里面被关闭为止
      
        RXDBUF[uart_i]=SBUF; //接收一个字节数据
        uart_i++;    //RXDBUF从0号元素不停往下存取数据   
    }
}

 

下一讲我们再教大家如何实践操作,代码可以了解了解意思先。

 


您可以捐助,支持我们的公益事业。

1元 10元 50元





认证码: 验证码,看不清楚?请点击刷新验证码 必填



414 次浏览
3次