找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 8498|回复: 0
打印 上一主题 下一主题
收起左侧

双通道电压采集电路设计 完整的单片机课程设计下载带程序

[复制链接]
跳转到指定楼层
楼主
附件里面有设计任务,根据附件原理图改正里面的程序。完整的word文档请在本文最后下载附件,下面是部分内容预览:




目     录
一、原理 ……………………………………………………………………………………………1
二、硬件介绍 ………………………………………………………………………………………1
   2.1 单片机介绍…………………………………………………………………………………1
2.2 AT24C02I2C数据存储 ……………………………………………………………………2
   2.3 PCF8591 A/D转换 …………………………………………………………………………2
   2.4 LCD1602 液晶显示模块……………………………………………………………………3
   硬件连接及工作过程简介………………………………………………………………………4
   3.1 硬件连接……………………………………………………………………………………4
   3.2 工作过程简介………………………………………………………………………………5
三、工作流程图 ……………………………………………………………………………………6
四、程序清单 ………………………………………………………………………………………7
五、PCB图 …………………………………………………………………………………………18
六、总结……………………………………………………………………………………………18
八、参考文献………………………………………………………………………………………19


一、原理本文所设计的是一种以单片机STC12C5A60S2(引脚排序及基本功能同AT89S51)作为主控芯片的双通道电压采集电路,将采集到的模拟信号电压值利用PCF8591来进行模数转换,PCF8591是一个单片集成、单独供电、低功耗、8-bitCMOS数据获取信号。其中PCF8591共有4个模拟输入、一个模拟输出和一个串行口I2C总线接口,通过PLCF8591将转换成的数字信号电压值传给液晶显示屏,并将所测电压值在LCD1602液晶显示屏中显示出来,而且要利用AT24C04设置电压上下限,在电压超出上下限时显示屏要有所表示.本文主要针对双通道的数据采集与显示,继而推广多通道数据采集的方法,重点分析介绍了基于AT89S51数字电压采集的硬件设计和软件设计。


图3-1硬件连接原理图




SDA、SCL是IIC数据线和时钟线,可连接于单片机任何IO管脚在。在本次设计中通过芯片AT24C02来编写所测电压的上下限值。
我们这次试验所测得电压是直流电压0至5伏是模拟信号由于我们所测得电压要在 LCD1602 液晶显示屏中显示出来,但是LCD1602液晶显示屏接受和表达数字信号,所以要用PCF8591 A/D转换芯片把采集的模拟信号电压值转换成数值信号,经过AT89S51再把信号传给LCD液晶显示屏,最终在液晶显示屏上显示出电压值。如果采集的电压值在设定的电压值之间,则电压值能在液晶显示屏上显示出电压,如果超出了则在液晶显示屏上有所表示。在此次试验中因为是双通道采集电压,所以我们测两条电路的电压,我在这两条电路上各串联了一个滑动变阻器,用来改变所测的电压值。

五、程序清单
  1. #include "common.h"

  2. float v_low = 2.8;  //电压范围下限
  3. float v_high = 4.5;  //电压范围上限
  4. uchar BUF[2];  //从at24c04中读取的数据存储在这
  5. uchar w_buf[2]; //写入at24c04中的数据存在这里

  6. //延时函数大约1ms
  7. void delay_ms(int ms)
  8. {
  9.         int i;
  10.         while(ms--)
  11.         {
  12.         for(i=0;i<100;i++);
  13.         }
  14. }

  15. //连续写入两个数据
  16. void AT24C04_Write(uchar dat_1,uchar dat_2)
  17. {
  18.         AT24C04_Start(); //起始信号
  19.         AT24C04_SendByte(0xa0); //发送设备地址+写信号
  20.         AT24C04_SendByte(0x00); //发送存储单元地址
  21.         AT24C04_SendByte(dat_1); //写入数据1
  22.         AT24C04_SendByte(dat_2); //写入数据2
  23.         AT24C04_Stop(); //停止信号
  24. }

  25. //连续读出两个数据
  26. void AT24C04_ReadPage()
  27. {
  28. uchar i;
  29. AT24C04_Start(); //起始信号
  30. AT24C04_SendByte(0xa0); //发送设备地址+写信号
  31. AT24C04_SendByte(0x00); //发送存储单元地址
  32. AT24C04_Start(); //起始信号
  33. AT24C04_SendByte(0xa1); //发送设备地址+读信号
  34. for (i=0; i<2; i++)
  35. {
  36. BUF[i] = AT24C04_RecvByte();
  37. if (i == 1)
  38. {
  39. AT24C04_SendACK(1); //最后一个数据需要会NAK
  40. }
  41. else
  42. {
  43. AT24C04_SendACK(0); //回应ACK
  44. }
  45. }
  46. AT24C04_Stop(); //停止信号
  47. }

  48. //LCD显示数字处理函数
  49. void show_num(uchar num,uchar flag)
  50. {
  51.         uchar temp[3],i;
  52.         float re_val;  //电压实际值
  53.         float read_low,read_h; //将从AT24C04中读取电压范围数据转化成对应小数形式

  54.         re_val = num/255.0*5.0; //实际电压计算公式

  55.         num = re_val*10; //保留一位小数

  56.          AT24C04_ReadPage();//从AT24C04中读取电压范围数据,存储在BUF数组中
  57.                                           // BUF[0]表示下限,BUF[1]表示上限
  58.         //存入AT24C04中的数据是uchar unsigned型,在这里转化成float型
  59.         read_low =         BUF[0]/10.0;
  60.         read_h  = BUF[1]/10.0;
  61.         if(re_val >= read_low && re_val <= read_h) //如果电压在范围内
  62.            {
  63.                 temp[0] = num/10 + '0';
  64.                 temp[1] = '.';
  65.                 temp[2] = num%10 + '0';
  66.                 }else if(re_val < read_low ) //如果低于设定值
  67.            {
  68.                    temp[0] = 'L';
  69.                 temp[1] = 'O';
  70.                 temp[2] = 'W';           
  71.            }else if(re_val > read_h ) //如果高于设定值
  72.            {
  73.                    temp[0] = 'H';
  74.                 temp[1] = 'I';
  75.                 temp[2] = 'G';         
  76.                 }
  77.         
  78.   for(i=0;i<3;i++)                 //用液晶显示结果
  79.   {
  80.            LCD_Manifest(flag,3+i,temp[i]);
  81.   }

  82.         //显示电压范围的上限
  83.     num = read_h * 10;
  84.         temp[0] = num/10 + '0';
  85.         temp[1] = '.';
  86.         temp[2] = num%10 + '0';
  87.   for(i=0;i<3;i++)                 //用液晶显示结果
  88.   {
  89.           LCD_Manifest(1,12+i,temp[i]);
  90.   }

  91.    //显示电压范围的下限
  92.     num = read_low * 10;
  93.         temp[0] = num/10 + '0';
  94.         temp[1] = '.';
  95.         temp[2] = num%10 + '0';
  96.   for(i=0;i<3;i++)                 //用液晶显示结果
  97.   {
  98.           LCD_Manifest(2,12+i,temp[i]);
  99.   }
  100. }

  101. //主函数
  102. void main()
  103. {
  104.         uchar val;        //电压值

  105.         LCD_Init();          //液晶初始化
  106.         IIC_Init();         //I2C总线初始化

  107.          w_buf[0] =         v_low*10;
  108.         w_buf[1] =         v_high*10;
  109.         AT24C04_Write(w_buf[0],w_buf[1]); //将数据写入AT24C04

  110.         LCD_Manifest(1,0,'V');        //在lcd1602中显示字符
  111.         LCD_Manifest(1,1,'1');
  112.         LCD_Manifest(1,2,':');
  113.         LCD_Manifest(2,0,'V');
  114.         LCD_Manifest(2,1,'2');
  115.         LCD_Manifest(2,2,':');

  116.         LCD_Manifest(1,10,'H');
  117.         LCD_Manifest(1,11,':');
  118.         LCD_Manifest(2,10,'L');
  119.         LCD_Manifest(2,11,':');


  120.         while(1)
  121.         {
  122.                 //按键检测程序
  123.                 if(key_1 == 0)           //调节范围的下限
  124.                 {
  125.                    delay_ms(100); //延时消除抖动
  126.                    if(key_1 == 0)
  127.                    {
  128.                   //         v_low = v_low - 0.1;
  129.                         // if(v_low < 0.0)
  130.                          //  v_low = v_high;

  131.                         w_buf[0] =         v_low*10;
  132.                         w_buf[1] =         v_high*10;
  133.                         AT24C04_Write(w_buf[0],w_buf[1]); //将数据写入AT24C04
  134.                    }
  135.                 }
  136.                 if(key_2 == 0)                //调节范围的上限
  137.                 {
  138.                    delay_ms(100); //延时消除抖动
  139.                    if(key_2 == 0)
  140.                    {
  141.                             v_high = v_high + 0.1;
  142.                          if(v_high > 5.0)
  143.                            v_high = v_low;

  144.                         w_buf[0] =         v_low*10;
  145.                         w_buf[1] =         v_high*10;
  146.                         AT24C04_Write(w_buf[0],w_buf[1]); //将数据写入AT24C04
  147.                    }
  148.                 }


  149.            ADC_Send_Byte(0x90,0x41);//通道一ADC转换         
  150.            val=ADC_Receive_Byte(0x90);        //读取AD转换的值
  151.            show_num(val,1);
  152.            ADC_Send_Byte(0x90,0x42);//通道二ADC转换         
  153.            val=ADC_Receive_Byte(0x90);        //读取AD转换的值
  154.            show_num(val,2);


  155.         }
  156. }

  157. #include<iic.h>
  158. void IIC_Init()//总线初始化
  159. {
  160.   
  161.   SCL=1;
  162.   IIC_DELAY();
  163.   SDA=1;
  164.   IIC_DELAY();
  165. }
  166. void IIC_Start() //启动信号
  167. {

  168.   SDA=1;
  169.   IIC_DELAY();
  170.   SCL=1;
  171.   IIC_DELAY();
  172.   SDA=0;
  173.   IIC_DELAY();
  174. }
  175. void IIC_Stop()        //停止信号
  176. {
  177.   
  178.    SDA=0;
  179.    IIC_DELAY();
  180.    SCL=1;
  181.    IIC_DELAY();
  182.    SDA=1;
  183.    IIC_DELAY();
  184. }
  185. void IIC_Ack() //应答信号
  186. {
  187.    unsigned char i;
  188.    SCL=1;
  189.    IIC_DELAY();;
  190.    while((SDA==1)&&(i<255))i++;
  191.    SCL=0;
  192.    IIC_DELAY();
  193. }
  194. void IIC_Write_Byte(u8 date)//写一个字节
  195. {
  196.    u8 i,temp;
  197.    temp=date;
  198.    for(i=0;i<8;i++)
  199.    {
  200.              temp=temp<<1;
  201.           SCL=0;
  202.           IIC_DELAY();
  203.           SDA=CY;//最高位将移入PSW寄存器的CY位中,然后将CY赋给SDA
  204.           IIC_DELAY();
  205.           SCL=1;
  206.           IIC_DELAY();
  207.    }
  208.    SCL=0;
  209.    IIC_DELAY();
  210.    SDA=1;
  211.    IIC_DELAY();
  212. }
  213. u8 IIC_Read_Byte()//读一个字节
  214. {
  215.    u8 i,temp;
  216.    for(i=0;i<8;i++)
  217.    {
  218.           SCL=0;
  219.           IIC_DELAY();
  220.           temp=(temp<<1)|SDA;
  221.           IIC_DELAY();
  222.           SCL=1;
  223.           IIC_DELAY();
  224.    }
  225.    SCL=0;
  226.    IIC_DELAY();
  227.    SDA=1;
  228.    IIC_DELAY();
  229.    return temp;
  230. }
  231. #include <reg51.h>
  232. #include"lcd1602.h"
  233. void delay(u8 z)//延时函数
  234. {
  235.   unsigned char x,y;
  236.   for(x=z;x>0;x--)
  237.     for(y=110;y>0;y--);
  238. }


  239. void LCD_Write_Cmd(u8 com )//液晶写命令
  240. {
  241.    LCD_RS=0;//选择写命令模式
  242.    LCD_dat=com;//将要写的命令字送到数据总线上
  243.    delay(1);
  244.    LCD_E=1;//使能端给一高脉冲,高脉冲有效
  245.    delay(1);
  246.    LCD_E=0;
  247. }
  248. void LCD_Write_Data(u8 date)//数据写命令
  249. {
  250.    LCD_RS=1;//选择写数据模式
  251.    LCD_dat=date;
  252.    delay(1);
  253.    LCD_E=1;//使能端给一高脉冲,高脉冲有效
  254.    delay(1);
  255.    LCD_E=0;
  256. }
  257. void LCD_Init()//液晶初始化
  258. {
  259.   
  260.    LCD_WR=0;
  261.    LCD_Write_Cmd(0x38);//设置16*2显示,5*7点阵,8位数据接口
  262.    LCD_Write_Cmd(0x0c);//设置开显示,不显示光标
  263.    LCD_Write_Cmd(0x06);//写一个字符后地址指针自动加1

  264.    LCD_Write_Cmd(0x80);//数据指针定位到第一行第一个字处

  265. }
  266. // 使得写命令和写数据通过一个函数完成
  267. void LCD_Manifest(u8 row,u8 add,u8 date)
  268. {
  269.          if(row==1)          LCD_Write_Cmd(0x80+add);
  270.         if(row==2)         LCD_Write_Cmd(0xc0+add);
  271.         LCD_Write_Data(date);
  272. }
  273. #include"iic.h"
  274. #include"pcf8591.h"

  275. //ADC发送字节[命令]数据函数
  276. void ADC_Send_Byte(u8 addr,u8 com)
  277. {
  278.    IIC_Start();              //启动总线
  279.    IIC_Write_Byte(addr);           //发送器件地址
  280.    IIC_Ack();
  281.    IIC_Write_Byte(com);             //发送数据
  282.    IIC_Ack();
  283.    IIC_Stop();              //结束总线
  284. }

  285. //ADC读字节数据函数
  286. u8 ADC_Receive_Byte(u8 addr)
  287. {
  288.    u8 c;
  289.    IIC_Start();          //启动总线
  290.    IIC_Write_Byte(addr+1);      //发送器件地址,如果是读则最低位为1即0x91
  291.    IIC_Ack();
  292.    c=IIC_Read_Byte();          //读取数据

  293.    IIC_Ack();          //发送非就答位
  294.    IIC_Stop();           //结束总线
  295.    return(c);
  296. }
  297. #include"AT24C04.h"
  298. void Delay5us()
  299. {
  300. BYTE n = 4;
  301. while (n--)
  302. {
  303. _nop_();
  304. _nop_();
  305. }
  306. }
  307. void Delay5ms()
  308. {
  309. WORD n = 2500;
  310. while (n--)
  311. {
  312. _nop_();
  313. _nop_();
  314. _nop_();
  315. _nop_();
  316. _nop_();
  317. }
  318. }
  319. void AT24C04_Start()
  320. {
  321. AT_SDA = 1; //拉高数据线
  322. AT_SCL = 1; //拉高时钟线
  323. Delay5us(); //延时
  324. AT_SDA = 0; //产生下降沿
  325. Delay5us(); //延时
  326. AT_SCL = 0; //拉低时钟线
  327. }
  328. void AT24C04_Stop()
  329. {
  330. AT_SDA = 0; //拉低数据线
  331. AT_SCL = 1; //拉高时钟线
  332. Delay5us(); //延时
  333. AT_SDA = 1; //产生上升沿
  334. Delay5us(); //延时
  335. }
  336. void AT24C04_SendACK(bit ack)
  337. {
  338. AT_SDA = ack; //写应答信号
  339. AT_SCL = 1; //拉高时钟线
  340. Delay5us(); //延时
  341. AT_SCL = 0; //拉低时钟线
  342. Delay5us(); //延时
  343. }
  344. bit AT24C04_RecvACK()
  345. {
  346. AT_SCL = 1; //拉高时钟线
  347. Delay5us(); //延时
  348. CY = AT_SDA; //读应答信号
  349. AT_SCL = 0; //拉低时钟线
  350. Delay5us(); //延时
  351. return CY;
  352. }
  353. void AT24C04_SendByte(BYTE dat)
  354. {
  355. BYTE i;
  356. for (i=0; i<8; i++) //8位计数器
  357. {
  358. dat <<= 1; //移出数据的最高位
  359. AT_SDA = CY; //送数据口
  360. AT_SCL = 1; //拉高时钟线
  361. Delay5us(); //延时
  362. AT_SCL = 0; //拉低时钟线
  363. Delay5us(); //延时
  364. }
  365. AT24C04_RecvACK();
  366. }
  367. BYTE AT24C04_RecvByte()
  368. {
  369. BYTE i;
  370. BYTE dat = 0;
  371. AT_SDA = 1; //使能内部上拉,准备读取数据
  372. for (i=0; i<8; i++) //8位计数器
  373. {
  374. dat <<= 1;
  375. AT_SCL = 1; //拉高时钟线
  376. Delay5us(); //延时
  377. dat |= AT_SDA; //读数据
  378. AT_SCL = 0; //拉低时钟线
  379. Delay5us(); //延时
  380. }
  381. return dat;
  382. }

复制代码

七、总结
本次课程设计,我学到了很多的东西,对汇编语言程序设计加深了理解和掌握,我们这次设计的要求主要是熟悉微型计算机技术中所学到的几中芯片的使用,掌握它们的初始化编程,并学会他们的基本应用,这是一个比较全面的设计,和以前的实验相比较复杂很多,一方面要求我们设计硬件电路,另一方面需要对设计的硬件电路进行编程,这就需要软件和硬件相结合,具体的电路具体分析,在此次设计中还用到了AT24C02芯片,在此次之前我们从没用到这个芯片,对这个芯片非常陌生,而且网上也没有什么系统的解释,和AT89S51的管脚的连线也不清楚,但是最后通过不停地调试,最后还是被解决了。通过这次课程设计后,我对学硬件有了一定的体会,它要求我们认真仔细的做好每个环节,对所有的问题要考虑周全,它和软件不同,对具体的电路要编写适合该电路的程序,电路一改,程序也要跟着改,从中我也学会了很多实际应用的知识。通过本次设计,我才发现我所学的东西的肤浅,知识面的不牢固,在选择方案的时候,产生模棱两可的情况。经常是边设计边复习学习过的课本知识。这些使我知道“纸上谈兵”是绝对不行,只有通过实践,自己亲自动手试一试,才能发现自己知识的缺陷,才能更好的与所学的知识相结合。在课程设计中我们每个人都能学到很多,有很大的收获。做任何事情所要有的态度和心态,首先我明白了做学问要一丝不苟,对于出现的任何问题和偏差都不要轻视,要通过正确的途径去解决,在做事情的过程中要有耐心和毅力,不要一遇到困难就打退堂鼓,只要坚持下去就可以找到思路去解决问题的。在工作中要学会与人合作的态度,认真听取别人的意见,这样做起事情来就可以事倍功半。

课程设计任务书下载: 任务书10----双通道电压采集电路设计(LCD显示).doc (36.5 KB, 下载次数: 19)
完整的设计文档下载: 单片机课设.doc (214.5 KB, 下载次数: 49)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏2 分享淘帖 顶 踩
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表