STC12C5A08S2 AD应用 电压表
jrcsh2010/10/10电子技术 IP:广东
代码很乱,调试的也没删除, 不过这个是包能用的  精度在 0.1V 左右 0.01的数据在乱飞机   (没有进行 电压校正我用的电源电压在 5.2V)  

这一个月~~~ 整STC 的AD~~~  实在无语  
  
  
过程中在网上找到了2个 参考的程序 但~~~有些问题的  特别是"东哥"的哪个 “补充”了不少定义上去后程序才能运行起来,但是1602还是出不来乱飞机的
  
  还是看着老摇的特色PDF 磨出来

数码管显示部分不通用  自己吧自己板上的复制过来 ok  


  调试的方法   是 用串口吧 AD 的数据 发送出来  以0.5V 一个档  记录AD 数据 计算出规律  

  老摇的~~~ 方法我没能~~~~整出花样来~~~只好用这个土方法 就这样~~~混过去了

   不要问我要电路 从代码中猜电路的接法  我也是用通用的电路搭的
  
    
   M0 阿快快出来~~~~~   (但愿M0~不难开发)
  
   愿这个~~~文档能带领您走出 STC的~~~恶梦  

   在BS 一下 STC 1T 的MCU 接着1602 不能进行下载程序
  

#include <absacc.h>
#include <intrins.h>
#include <reg52.h>

   sfr P1ASF = 0x9D;
   sfr ADC_CONTR   = 0xBC;
   sfr ADC_RES   = 0xBD;
   sfr ADC_RESL   = 0xBE;
   sfr AUXR1 = 0xA2;    
   sfr P1M1   = 0x91 ;
   sfr P1M0   = 0x92 ;
   sfr   smdis = 0x80;

    sbit led = P3^7;
    sbit weishuang0 = P2^4;   //数码管第一个位选
    sbit weishuang1 = P2^5;   //数码管第二个位选
    sbit weishuang2 = P2^6;   //数码管第三个位选
    sbit weishuang3 = P2^7; //数码管第三个位选
    sbit d5=P0^5;
//以下选择 ADC 转换速率,只能选择其中一种
//            SPEED1 SPEED0 A/D转换所需时间
//#define AD_SPEED   0x60 // 0110,0000   1      1   70 个时钟周期转换一次,  
// CPU工作频率21MHz时 A/D转换速度约 300KHz
#define AD_SPEED   0x40 //0100,0000   1      0     140 个时钟周期转换一次
//#define AD_SPEED   0x20 //0010,0000   0      1     280 个时钟周期转换一次
//#define AD_SPEED   0x00 //0000,0000   0      0     420 个时钟周期转换一次

unsigned char dp=0x10;
#define uchar unsigned char
#define uint unsigned int
uchar code comm[]={'0','1','2','3','4','5','6','7','8','9'};
unsigned char code md[]={0x28,0xeb,0x32,0xa2,0xe1,0xa4,0x24,0xea,0x20,0xa0,0xff};  
                          //0,1,2,3,4,5,6,7,8,9,1.,2.,3.,4.,5.,6.,7.,8.,9., -,灭 ,
unsigned char shu0,shu1,shu2,shu3,tc=0;
unsigned char vccreal;    //设定实际的电压值
unsigned int vv,vv1,vv2,vv3;  
//bit FLAG1=0;//是否检测到DS18B20标志位
uint temp;

    void int_t0() interrupt 1
          {
            TR0 = 0;
               tc++;
     if(tc>=4) {tc=0;}    
             TH0 = 0xee;
             TL0 = 0x00;              
              if(tc==0) {  weishuang3 = 1;  smdis=md[shu0];  weishuang0 = 0;}        
         else if(tc==1) {  weishuang0 = 1;  smdis=md[shu1];  weishuang1 = 0;}            
         else if(tc==2) {  weishuang1 = 1;  smdis=md[shu2];  weishuang2 = 0;}              
         else if(tc==3) {  weishuang2 = 1;  smdis=md[shu3];d5=0;  weishuang3 = 0;}  
           TR0 = 1;

             /*

P0=table[ww]; // ok
w1=0;

P0=table[ww];

P0=table[ww]; // ok
w3=0;

P0=table[ww];
w4=0;

*/
      
          }
    void init_mcu()
          {
             TMOD=0x21; //定时器0方式1
             TH0 = 0xee;//0xa6; //25ms定时常数
             TL0 = 0x00;  
          EA = 1;
             ET0 = 1;
             TR0 = 1;
             tc = 0;
              //TMOD=0x20;//设置定时器1为工作方式2
//TH1=0xfd;
// TL1=0xfd;
// TR1=1;
// REN=1;
// SM0=0;
// SM1=1;
// EA=1;
// ES=1;

    //
             shu0=shu1=shu2=shu3=0;
           }



void delayt(unsigned char t)        // 延时函数
        {
          unsigned int n;
            while(t--)  
              {  
                 n = 6000;
                 while(--n);
               }
         }


//---------------------------------------------------------------------
unsigned int get_AD_result(unsigned char channel)
{
    unsigned int AD_RESULT=0;            //存储 A/D 转换结果
// float temp1;
//float temp2;
//uint temp3;
    ADC_RES = 0;
    ADC_RESL = 0;

    channel &= 0x07;                //0000,0111 清0高5位


    ADC_CONTR = AD_SPEED;
    _nop_(); _nop_();
    ADC_CONTR |= channel;           //选择 A/D 当前通道
    _nop_();   _nop_();
    ADC_CONTR |= 0x80;              //启动 A/D 电源
      led   = 0;
delayt(3);                       //使输入电压达到稳定
    ADC_CONTR |= 0x08;              //0000,1000 令 ADCS = 1, 启动A/D转换,  

    
    while (!(ADC_CONTR & 0x10) ) ;       //等待A/D转换结束   //0001,0000 测试A/D转换结束否
    
    ADC_CONTR &= 0xE7;              //1111,0111 清 ADC_FLAG 位, 关闭A/D转换,  
   led   = 1;

AD_RESULT = ADC_RES*4+ ADC_RESL ; // &0x03

/*
temp=ADC_RES;//高8位
temp1=temp;
temp=temp1/256*5*10000;//高8位计算

temp3=ADC_RESL;//低2位
temp2=temp3;
temp3=temp2/256*5*10000;//低2位计算

AD_RESULT=temp+temp3;//高8位的计算值+低2位的计算值=实际值




-----------------------------------
vv=ADC_RES;//高8位
vv1=vv;
vv=vv1/256*5*10000;//高8位计算

vv3=ADC_RESL;//低2位
vv2=vv3;
vv3=vv2/256*5*10000;//低2位计算

vv=vv+vv3;//高8位的计算值+低2位的计算值=实际值


*/


return (AD_RESULT);  

  //return (vv);               //返回 A/D 高 8 位转换结果
}


void main()    //主函数
{
    unsigned char  m; //d0,d1,d2,d3,d01,d11,d21,d31,d02,d12,d22,d32,d03,d13,d23,d33;
unsigned int     mmm;
//unsigned   int num2;

led   = 1;

   vccreal = 5;    //设定实际的电压值

    init_mcu();

     P1M1   = 0xff ;   //P1口设置为高阻
     P1M0   = 0x00 ;


P1ASF = 0x80;                   //0000,0010, 将 P1.7 置成模拟口
AUXR1 &= ~0x04;                 //0000,0100, 令 ADRJ=0
//    0: 10 位A/D 转换结果的高8 位放在ADC_RES 寄存器, 低2 位放在ADC_RESL 寄存器
    //    1: 10 位A/D 转换结果的最高2 位放在ADC_RES 寄存器的低2 位, 低8 位放在ADC_RESL 寄存器
   ADC_CONTR |= 0x80;              //1000,0000 打开 A/D 转换电源


while(1)
{  
    delayt(1);
for(m=1;m<4;m++)
{
   // P1M1   = 0xff ;  
  _nop_();   _nop_();   _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
   mmm = get_AD_result(0);//P1.7 为 A/D 当前通道, 测量并发送结果
  led   = 1;
     delayt(10);
  // P1M1   = 0x00;  
     _nop_();   _nop_();   _nop_(); _nop_(); _nop_(); _nop_();
mmm=(mmm+10)/2;
/*
if(m==1)
{
SBUF=comm[mmm/100%10];
while(!TI);
TI=0;

SBUF=comm[mmm/10%10];
while(!TI);
TI=0;

SBUF=comm[mmm%10];
while(!TI);
TI=0;

}

if(m==12)
{
SBUF=comm[mmm%10];
while(!TI);
TI=0;

SBUF=comm[mmm/10%10];
while(!TI);
TI=0;

SBUF=comm[mmm/100%10];
while(!TI);
TI=0;

SBUF=comm[mmm/1000%10];
while(!TI);
TI=0;
}

if(m==13)
{
SBUF=comm[mmm%10];
while(!TI);
TI=0;

SBUF=comm[mmm/10%10];
while(!TI);
TI=0;

SBUF=comm[mmm/100%10];
while(!TI);
TI=0;

SBUF=comm[mmm/1000%10];
while(!TI);
TI=0;

SBUF=comm[mmm/10000%10];
while(!TI);
TI=0;

SBUF=comm[mmm/100000%10];
while(!TI);
TI=0;
}



SBUF='A';//28;
while(!TI);
TI=0;
*/
d3=mmm/100%10;//显示百位


d2=mmm/10%10;//显示十位


d1=mmm%10;//显示个位

d0=3;

shu0=d0;   //显示            
shu1=d1;   //显示 小数点后一位    
shu2=d2;   //显示    小数点后两位  
shu3=d3;   //显示      小数点后三位    19;//
//d0=mmm/10;//显示小数点:0.001
//DispBuf[4]=mm%10;//显示小数点:0.0001



/*
   num1 = num;
   d0 = num1/256 ; //取出个位

   // num = num1-d0*1024;       //小数点后的数  
   num = num1%256;
        
   d1 = (num*10)/256; //取出 位
     num1 = (num*10)%256;
   // num1 = num*10-d1*1024;
      
   d2 = (num1*10)/256;

   //d3 = (num1*10 -d2*1024);//%10 ;
    num   = (num1*10)%256 ;
     d3   = (num *10)/256;

    d0=d0+10;    //加上小数点   个位

if(m==1)
{  

    d31=d3;   //显示            
    d21=d2;   //显示 小数点后一位    
    d11=d1;   //显示    小数点后两位  
    d01=d0;   //显示      小数点后三位    19;//
}
if(m==2)
{  

    d32=d3;   //显示            
    d22=d2;   //显示 小数点后一位    
    d12=d1;   //显示    小数点后两位  
    d02=d0;   //显示      小数点后三位    19;//
}
if(m==3)
{  

     d33=d3;   //显示            
     d23=d2;   //显示 小数点后一位    
     d13=d1;   //显示    小数点后两位  
     d03=d0;   //显示      小数点后三位    19;//
}
*/
          
}//for
/*
shu0=d31/3+d32/3+d33/3;   //显示            
shu1=d21/3+d22/3+d23/3;   //显示 小数点后一位    
shu2=d11/3+d12/3+d13/3;   //显示    小数点后两位  
shu3=d01/3+d02/3+d03/3;   //显示      小数点后三位    19;//

shu0=(d31+d32+d33)/3;   //显示            
shu1=(d21+d22+d23)/3;   //显示 小数点后一位    
shu2=(d11+d12+d13)/3;   //显示    小数点后两位  
shu3=(d01+d02+d03)/3;   //显示      小数点后三位    19;//
//shu0=0;   shu1=1; shu2=2;   shu3=3;
*/
}
}
来自:电子信息 / 电子技术
2
已屏蔽 原因:{{ notice.reason }}已屏蔽
{{notice.noticeContent}}
~~空空如也
warmonkey
14年4个月前 IP:未同步
261166
输入端应该至少使用RC 1阶低通滤波,并且让采样率至少是截止频率的20倍,如果能取100倍更好。
以防止未去除的高频分量被AD采样后,出现混叠现象,导致干扰无法完全滤除。
引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论

想参与大家的讨论?现在就 登录 或者 注册

所属专业
所属分类
上级专业
同级专业
jrcsh
学者 笔友
文章
491
回复
6730
学术分
4
2009/01/02注册,7年0个月前活动

.

主体类型:个人
所属领域:无
认证方式:邮箱
IP归属地:未同步
插入公式
评论控制
加载中...
文号:{{pid}}
投诉或举报
加载中...
{{tip}}
请选择违规类型:
{{reason.type}}

空空如也

加载中...
详情
详情
推送到专栏从专栏移除
设为匿名取消匿名
查看作者
回复
只看作者
加入收藏取消收藏
收藏
取消收藏
折叠回复
置顶取消置顶
评学术分
鼓励
设为精选取消精选
管理提醒
编辑
通过审核
评论控制
退修或删除
历史版本
违规记录
投诉或举报
加入黑名单移除黑名单
查看IP
{{format('YYYY/MM/DD HH:mm:ss', toc)}}