找回密码
 立即注册

QQ登录

只需一步,快速开始

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

51单片机学习:利用ADC0804模数转换器采集电压

[复制链接]
跳转到指定楼层
楼主
ID:161768 发表于 2017-2-9 22:17 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
电位器调节待检测电压值,在数码管上显示出来,
代码大多从书上搬过来的,书上例5.3.1要求前3个数码管显示AD转换后的8位数字量(即0~255)
我这里让前4个数码管显示具体电压值,比如1.352
  1.     #include <reg52.h>  
  2.     #include "MY51.H"  
  3.       
  4.     void initSMG()      //数码管初始化信息  
  5.     {  
  6.         //上电时,都为高电平  
  7.         P0=0xff;  
  8.         wela=open;  
  9.         P0=0xff;  
  10.         wela=lock;  
  11.       
  12.         P0=0;  
  13.         dula=open;  
  14.         P0=0;  
  15.         dula=lock;  
  16.     }  
  17.       
  18.     void ADC0804_csToLow()  //cs置低电平  
  19.     {  
  20.         wela=open;  //打开锁存器  
  21.         P0=0x7f;        //锁存器最高位送0,也就是CSAD置0  
  22.         wela=lock;  
  23.     }  
  24.       
  25.     void ADC0804_startConvert() //P3.6口是wr,由高到底,再拉高后,ad开始转换  
  26.     {  
  27.         adwr=high;  //虽然原本就是高的,但我们要养成好习惯,该是什么就是什么  
  28.         _nop_();  
  29.         adwr=low;   //wr置低后,过小会后AD内部开始执行转换,转换完成后INTR自动置低触发中断  
  30.         _nop_();  
  31.         adwr=high;  
  32.         //由于我们用的不是150pF电容,而是104pF,所以转换比较慢,在调用本函数后最好延时10毫秒以上  
  33.     }  
  34.       
  35.     uint8 ADC0804_readResult()  //读转换结果  
  36.     {  
  37.         uint8 result=0;  
  38.         //延时一会儿,转换就完成了,由于我们将INTR和CS都拉低了,直接操作RD后就可以读了  
  39.         P1=0xff;    //防止由于转换未完成原因引起的误读  
  40.         adrd=high;  
  41.         _nop_();  
  42.         adrd=low; //rd置低电平后数据总线P1口得到数据,并由led显示现象  
  43.         _nop_();  
  44.         result=P1;  
  45.         adrd=high;  
  46.         //读完以后,如果ad芯片不用了,就把cs拉高,注销片选  
  47.         return result;  
  48.     }   
  49.       
  50.     void show(uint8 value)   //基准电压是2.5V 为了计算方便扩大到2500  
  51.     {  
  52.         uint16 temp=value*(2500/255.0); //扩大到4位整数,小数点另外附加显示  
  53.         uint8 oneWela,twoWela,threeWela,fourWela; //oneWela是最左边的数码管  
  54.         oneWela=temp/1000;  
  55.         twoWela=temp%1000/100;  
  56.         threeWela=temp%100/10;  
  57.         fourWela=temp%10;  
  58.         displaySMG(oneWela,twoWela,threeWela,fourWela,dark,dark,dotTable[1]); //最左边的数码管显示小数点  
  59.     }  
  60.       
  61.     void main()  
  62.     {  
  63.       
  64.         uchar i=0;  
  65.         uchar adTemp=0;  
  66.         initSMG();          //数码管数据初始化  
  67.         ADC0804_csToLow();   //cs置低  
  68.         while(1)  
  69.         {  
  70.             ADC0804_startConvert(); //开始将电压数据转换成数字信号      
  71.             for(i=5;i>0;i--)   //主要是延时一段时间,让ad完成转换  
  72.             {  
  73.                 delayms(1);  
  74.                 show(adTemp);   //延时的时候,数码管继续动态显示   
  75.             }  
  76.             adTemp=ADC0804_readResult(); //读取数据  
  77.             show(adTemp);  
  78.         }  
  79.     }  
复制代码
  1.     #ifndef _MY51_H_  
  2.     #define _MY51_H_  
  3.     #include <math.h>  
  4.     #include <intrins.h>  
  5.       
  6.     typedef int                 int16  ;  
  7.     typedef int                 INT16  ;  
  8.     typedef unsigned int    uint16 ;  
  9.     typedef unsigned int    UINT16 ;  
  10.     typedef unsigned short  uint ;  
  11.     typedef unsigned short  UINT ;  
  12.     typedef unsigned short  word ;  
  13.     typedef unsigned short  WORD ;  
  14.     typedef unsigned long   uint32 ;  
  15.     typedef unsigned long   UINT32 ;  
  16.     typedef unsigned long   DWORD ;  
  17.     typedef unsigned long   dword ;  
  18.     typedef signed long    int32     ;  
  19.     typedef signed long    INT32  ;  
  20.     typedef float                   float32  ;  
  21.     typedef double              double64  ;  
  22.     typedef signed char     int8 ;  
  23.     typedef signed char         INT8 ;  
  24.     typedef unsigned char   byte ;  
  25.     typedef unsigned char    BYTE    ;      //WINDOWS的windef.h里面是这么定义的  
  26.     typedef unsigned char   uchar ;  
  27.     typedef unsigned char   UCHAR ;  
  28.     typedef unsigned char   UINT8 ;  
  29.     typedef unsigned char   uint8 ;  
  30.     typedef unsigned char   BOOL     ;      //windows中定义BOOL为int  
  31.     typedef unsigned char   bool     ;          //bool是c++的内置类型  
  32.       
  33.     #define TRUE     1  
  34.     #define true     1  
  35.     #define FALSE    0  
  36.     #define false    0  
  37.       
  38.     #define open     1    //open和close用于 标志打开和关闭状态  
  39.     #define OPEN     1  
  40.     #define close    0  
  41.     #define CLOSE    0  
  42.     #define lock     0  
  43.     #define start    1  
  44.     #define START    1  
  45.     #define stop     0  
  46.     #define STOP     0  
  47.     #define keyDown  0  
  48.     #define keyUp    1  
  49.     #define gnd       0  //接地  
  50.     #define GND       0  //接地  
  51.     #define high    1  //高电平  
  52.     #define low     0   //低电平  
  53.     #define yes     1  
  54.     #define YES     1  
  55.     #define no      0  
  56.     #define NO      0  
  57.       
  58.     sbit dula =P2^6;  //段选锁存器控制  控制笔段  
  59.     sbit wela =P2^7;  //位选锁存器控制  控制位置  
  60.       
  61.     #define led P1    //灯总线控制  
  62.     sbit led0=P1^0;   //8个led灯,阴极送低电平点亮  
  63.     sbit led1=P1^1;  
  64.     sbit led2=P1^2;  
  65.     sbit led3=P1^3;  
  66.     sbit led4=P1^4;  
  67.     sbit led5=P1^5;  
  68.     sbit led6=P1^6;  
  69.     sbit led7=P1^7;  
  70.       
  71.     sbit keyS2=P3^4;    //4个独立按键  
  72.     sbit keyS3=P3^5;  
  73.     sbit keyS4=P3^6;  
  74.     sbit keyS5=P3^7;  
  75.       
  76.     sbit csda=P3^2;  //DAC0832模数转换cs口  
  77.     sbit adwr=P3^6; //ADC0804这个同DAC0832  
  78.     sbit dawr=P3^6;  
  79.     sbit adrd=P3^7;  //ADC0804  
  80.     sbit beep=P2^3;  //蜂鸣器  
  81.     void displaySMG(uint8 one,uint8 two,uint8 three,uint8 four,uint8 five,uint8 six,uint8 dot);   
  82.     void delayms(uint16 ms);  
  83.     void T0_Work();  
  84.       
  85.     void delayms(uint16 ms)  //软延时函数  
  86.     {  
  87.         uint16 i,j;  
  88.         for(i=ms;i>0;i--)  
  89.         {  
  90.             for(j=113;j>0;j--)  
  91.             {}  
  92.         }  
  93.     }  
  94.     ///////////////////////////////////////////////////////////////////////////  
  95.     #define dark    0x11  //在段中,0x11是第17号元素,为0是低电平,数码管不亮  
  96.     #define dotDark 0xff //小数点全暗时  
  97.     uint8 code table[]= {           //0~F外加小数点和空输出的数码管编码  
  98.         0x3f , 0x06 , 0x5b , 0x4f , // 0 1 2 3  
  99.         0x66 , 0x6d , 0x7d , 0x07 , // 4 5 6 7  
  100.         0x7f , 0x6f , 0x77 , 0x7c , // 8 9 A B  
  101.         0x39 , 0x5e , 0x79 , 0x71 , // C D E F  
  102.         0x80 , 0x00 ,0x40           // . 空  负号    空时是第0x11号也就是第17号元素  
  103.      };  
  104.       
  105.     uint8 dotTable[]={         //小数点位置  
  106.         0xff ,                 //全暗  
  107.         0xfe , 0xfd , 0xfb ,   //1 2 3  
  108.         0xf7 , 0xef , 0xdf     //4 5 6                     
  109.     };  
  110.     /////////////////////////////////////////////////////////////////////////////  
  111.     uint8   TH0Cout=0 ;     //初值      
  112.     uint8   TL0Cout=0 ;      
  113.     uint16  T0IntCout=0;     //中断计数  
  114.     uint16  T0IntCountAll=0; //(N-1)/65536+1;  //总中断次数  
  115.     bool    bT0Delay=false;  //使用延时函数标志,初始未用  
  116.     bool    bT0Over=false;   //中断处理函数执行结果之一  
  117.       
  118.     void startT0(uint32 ms)  //开启定时器  
  119.     {     
  120.         float32     t=ms/1000.0;                 //定时时间  
  121.         double64    fox =11.0592*(pow(10,6));   //晶振频率  
  122.         uint32      N=(t*fox)/12 ;               //定时器总计数值  
  123.       
  124.         TH0Cout =(65536-N%65536)/256;        //装入计时值零头计数初值  
  125.         TL0Cout =(65536-N%65536)%256;  
  126.         T0IntCountAll=(N-1)/65536+1;             //总中断次数  
  127.         TMOD=TMOD | 0x01;                        //设置定时器0的工作方式为1  
  128.          
  129.         EA =open;   //打开总中断  
  130.         ET0=open;   //打开定时器中断  
  131.       
  132.         TH0=TH0Cout;  //定时器装入初值  
  133.         TL0=TL0Cout;  
  134.         TR0=start;   //启动定时器  
  135.     }  
  136.       
  137.     void delayT0(uint32 ms)     //硬延时函数,自己乱写的不好用,求指点  
  138.     {  
  139.         startT0(ms);                //启动定时器  
  140.         bT0Delay=true;          //告诉T0定时器,起用延时模式  
  141.         while(bT0Over==false);  //时间没到的话继续检测  
  142.         bT0Over=false;              //时间到了,让标志复位  
  143.     }  
  144.       
  145.     void T0_times() interrupt 1 //T0定时器中断函数  
  146.     {  
  147.         T0IntCout++;  
  148.         if(T0IntCout==T0IntCountAll)  //达到总中断次数值  
  149.         {     
  150.             T0IntCout=0;         //中断次数清零,重新计时  
  151.             bT0Over=true;     //时间真的到了  
  152.             if(bT0Delay)        //本次中断是用来延时的吗  
  153.             {  
  154.                 TR0=stop;        //如果是由延时函数开启T0的话,关闭T0  
  155.                 return;  
  156.             }  
  157.       
  158.             TH0=TH0Cout;        //循环定时的话要重装初值,每次定时1秒,重装一次  
  159.             TL0=TL0Cout;  
  160.             T0_Work();          //工作函数  
  161.         }  
  162.     }  
  163.       
  164.     ////////////////////////////////////////////////////////////////////////////////  
  165.     void displaySMG(uint8 oneWela,uint8 twoWela,uint8 threeWela,uint8 fourWela,uint8 fiveWela,uint8 sixWela,uint8 dot)  
  166.     {     
  167.         //控制6位数码管显示函数,不显示的位用参数dark,保留ADC0804的片选信号  
  168.         uint8 csadState=0x80&P0;                //提取最高位,即ADC0804的片选信号  
  169.         uint8 tempP0=((csadState==0)?0x7f:0xff); //数码管位选初始信号,阴极全置高电平  
  170.         P0=tempP0;      //0x7f表示数码管不亮,同时ADC0804片选有效  
  171.         wela=1;         //注:wela和dula上电默认为1  
  172.         P0=tempP0;  
  173.         wela=0;  
  174.       
  175.         P0=0;               //由于数码管是共阴极的,阳极送低电平,灯不亮,防止灯误亮  
  176.         dula=1;  
  177.         P0=0;  
  178.         dula=0;             //段选数据清空并锁定  
  179.     //////////////////////////oneWela  
  180.         {  //消除叠影,数码管阴极置高电平,并锁存  
  181.             P0=tempP0;  
  182.             wela=1;           
  183.             P0=tempP0;  
  184.             wela=0;  
  185.         }  
  186.         P0=0;           //低电平送到数码管阳极,避免数码管误亮  
  187.         dula=1;  
  188.         P0=table[oneWela]|((0x01&dot)?0x00:0x80);   //送段数据,叠加小数点的显示  
  189.         dula=0;  
  190.          
  191.       
  192.         P0=tempP0;          //送位数据前关闭所有显示,并保持csad信号  
  193.         wela=1;  
  194.         P0=tempP0 & 0xfe;   //0111 1110最高位是AD片选,低6位是数码管位选,低电平有效  
  195.         wela=0;  
  196.         delayms(2);  
  197.       
  198.     /////////////////////////twoWela  
  199.         {  //消除叠影  
  200.             P0=tempP0;  
  201.             wela=1;           
  202.             P0=tempP0;  
  203.             wela=0;  
  204.         }  
  205.         P0=0;  
  206.         dula=1;  
  207.         P0=table[twoWela]|((0x02&dot)?0x00:0x80);  
  208.         dula=0;  
  209.          
  210.         P0=tempP0;  
  211.         wela=1;  
  212.         P0=tempP0 & 0xfd;    //0111 1101  
  213.         wela=0;  
  214.         delayms(2);  
  215.       
  216.     /////////////////////////threeWela  
  217.         {  //消除叠影  
  218.             P0=tempP0;  
  219.             wela=1;           
  220.             P0=tempP0;  
  221.             wela=0;  
  222.         }  
  223.         P0=0;  
  224.         dula=1;  
  225.         P0=table[threeWela]|((0x04&dot)?0x00:0x80);  
  226.         dula=0;  
  227.       
  228.         P0=tempP0;  
  229.         wela=1;  
  230.         P0=tempP0 & 0xfb;    //0111 1011  
  231.         wela=0;  
  232.         delayms(2);  
  233.       
  234.     /////////////////////////fourWela  
  235.         {  //消除叠影  
  236.             P0=tempP0;  
  237.             wela=1;           
  238.             P0=tempP0;  
  239.             wela=0;  
  240.         }  
  241.         P0=0;  
  242.         dula=1;  
  243.         P0=table[fourWela]|((0x08&dot)?0x00:0x80);  
  244.         dula=0;  
  245.       
  246.         P0=tempP0;  
  247.         wela=1;  
  248.         P0=tempP0 & 0xf7;   //0111 0111  
  249.         wela=0;  
  250.         delayms(2);  
  251.       
  252.     /////////////////////////fiveWela  
  253.         {  //消除叠影  
  254.             P0=tempP0;  
  255.             wela=1;           
  256.             P0=tempP0;  
  257.             wela=0;  
  258.         }  
  259.         P0=0;  
  260.         dula=1;  
  261.         P0=table[fiveWela]|((0x10&dot)?0x00:0x80);  
  262.         dula=0;  
  263.       
  264.         P0=tempP0;  
  265.         wela=1;  
  266.         P0=tempP0 & 0xef;       //0110 1111  
  267.         wela=0;  
  268.         delayms(2);  
  269.       
  270.     /////////////////////////sixWela  
  271.         {  //消除叠影  
  272.             P0=tempP0;  
  273.             wela=1;           
  274.             P0=tempP0;  
  275.             wela=0;  
  276.         }  
  277.         P0=0;  
  278.         dula=1;  
  279.         P0=table[sixWela]|((0x20&dot)?0x00:0x80);  
  280.         dula=0;  
  281.       
  282.         P0=tempP0;  
  283.         wela=1;  
  284.         P0=tempP0 & 0xdf;   //0101 1111  
  285.         wela=0;  
  286.         delayms(2);  
  287.     }  
  288.       
  289.     #endif  
复制代码




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

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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