实验原理图:
实验源代码:
/**************************************************************************************/ /*单片机采用STC89C52,晶振为12MHz。P1.0与DS18B20的数据端相连*/ /*P0为数码管的字段码口,P2口为位选码口*/ /*P3.2与按键0相连,P3.3与按键1相连*/ /*MAXNUM是单总线上最大可扫描DS18B20个数*/ /*flag为温度的正负号标志单元,flag为"1"时表示温度值为负值,为"0"时表示温度值为正值*/ /*温度测量范围为(-55.0~99.9摄氏度)。*/ /*变量cc中保存计算出的温度值的整数部分,xs保存计算出的温度值的小数部分的第一位*/ /*num记录当前单总线上DS18B20的个数*/ /*z是按键标志位,z为1时表明有按键按下*/ /*a,b分别是按键0和按键1的记录变量*/ /*m是轮换显示各DS18B20温度值和序列号的全局标志变量*/ /*二维数组ID用于记录各DS18B20的ROM序列号*/ /*数组disbuffer是LCD显示缓存数组*/ /*数组RomID_temp用于匹配DS18B20时临时记录要匹配DS18B20的序列号*/ /*共用体temp用于存放从DS18B20读入的数据*/ /**************************************************************************************/ #include <reg51.h> #include <intrins.h> #define uchar unsigned char #define uint unsigned int #define MAXNUM 4 //宏定义单总线上最大可扫描DS18B20个数 sbit DS=P2^2; //用P1.0口作为各DS18B20与单片机的I/O口 sbit Key0=P3^2; //P3.2用作按键0的输入,采用外部中断方式获取按键信号 sbit Key1=P3^3; //P3.3用作按键1的输入,采用外部中断方式获取按键信号 sbit dula=P2^6; sbit wela=P2^7; //unsigned char code chocode[]={0xbf,0x86,0xdb,0xcf,0xe6,0xed,0xfd,0x87,0xff,0xef}; union{ //定义共用体temp用于存放从DS18B20读入的数据,Keil C51采用大端格式,c[1]是高地址. uchar c[2]; //其中存放读取温度数值的低字节;c[0]是低地址,其中存放读取温度数值的高字节. uint x; //这样根据大端格式的要求,即数据的高字节存储在低地址中,数据的低字节存放在高地址中, }temp; //x便刚好是读出的温度数值. uchar idata flag; //温度的正负号标志,flag为"1"时表示温度值为负值,为"0"时表示温度值为正值 uint cc,xs; //变量cc中保存计算出的温度值的整数部分,xs保存计算出的温度值的小数部分的第一位 uchar idata disbuffer[6]; //LCD显示缓存数组 uchar idata ID[4][8]={0}; //用于记录各DS18B20的ROM序列号 uchar idata RomID_temp[8]; //匹配DS18B20时临时记录要匹配DS18B20的序列号 uchar m=0; //m是轮换显示各DS18B20温度值和序列号的全局标志变量 uchar num=0; //num记录当前单总线上DS18B20的个数 uchar z=0; //z是按键标志位,z为1时表明有按键按下 uchar a=0; //a是按键0的记录变量 uchar b=0; //b是按键1的记录变量 void delay(uint i) //延时i*9.62us { uint j; for(j=i;j>0;j--); } void delay_ms(uchar i) //延时(j*2+1+2)*i+5 个机器周期 { uchar j; //12MHz时,延时 0.5*i ms do{j=248; do{j--;}while(j); i--; }while(i); } void delay_2us(uchar i) //延时 2*i+5 us { while(--i); } uchar DS_init(void) //18B20复位,初始化函数 { uchar presence; DS=0; delay_2us(250); //根据DS18B20的复位时序.先把总线拉低555us DS=1; delay_2us(30); //再释放总线,65us后读取DS18B20发出的信号 presence=DS; delay_2us(250); //如果复位成功,则presence的值为0;否则为1 return (presence); //返回0则初始化成功,否则失败 } uchar read_byte(void) //读1字节 { uchar i,j,dat=0; for(i=1;i<=8;i++) //作8个循环,读出的8位组成一个字节 {DS=0; _nop_(); //先将总线拉低1us, DS=1; delay_2us(2); //再释放总线,产生读起始信号,延迟9us后读取总线上的DS18B20发出的值 j=DS; delay_2us(30); //一位读完后,延迟65us后读下一位 dat=(j<<7)|(dat>>1); //读出的数据最低位在一个字节的最低位,这样刚好一个字节在DAT里 } return(dat); } uchar read_2bit(void) //读2位 { uchar i=0,j=0; DS=0; _nop_(); //先将总线拉低1us, DS=1; delay_2us(2); //再释放总线,产生读起始信号,延迟9us后读取总线上的DS18B20发出的值 j=DS; delay_2us(30); //一位读完后,延迟65us后读下一位 DS=0; _nop_(); DS=1; delay_2us(2); i=DS; delay_2us(30); i=j*2+i; //将读出的两位放到变量i中,其中第一个读出的位处于i的第1位;而第二个读出的位处于i的第0位 return(i); } void write_byte(uchar dat) //写1字节 { uchar i; for(i=0;i<8;i++) //作8个循环,写入的8位组成一个字节 {DS=0; //先将总线拉低 DS = dat&0x01; //向总线上放入要写的值 delay_2us(50); //延迟105us,以使DS18B20能采样到要写入的值 DS = 1; //释放总线,准备写入下一位 dat>>=1; //将要写的下一位移到dat的最低位 } } void write_bit(bit dat) //写1位 { DS=0; //先将总线拉低 DS=dat; //向总线上放入要写的值 delay_2us(50); //延迟105us,以使DS18B20能采样到要写入的值 DS = 1; //释放总线 } void display_ROMID(void) //定义显示序列号的函数 { uchar i,p,t,k; uint q; uchar disbuffer_rom[8]; uchar codevalue[]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xf8,0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e,0xff,0xbf}; //共阳极的字段码 uchar chocode[]={0xf7,0xfb,0xfd,0xfe,0x7f,0xbf,0xdf,0xef}; //位选码表 z=0; //z归零,表明没有按键按下 for(q=0;q<500;q++) //显示各DS18B20的序号 { if(z==1) //如果z为1,有按键按下,表明要显示不同的内容,此时从这个显示循环中跳出 break; t=chocode[7]; //取当前的位选码 P2=t; //送出位选码 t=codevalue[m+10]; //查得显示字符的字段码 P0=t; //送出字段码 delay(100); } P2=0xFF; //关断LCD一段时间,产生闪屏效果 for(q=0;q<500;q++) { if(z==1) //如果z为1,有按键按下,表明要显示不同的内容,此时从这个循环中跳出 break; delay(100); } for(k=0;k<8;k=k+4) //依次显示序列号的低32位或高32位 { if(z==1) //如果z为1,有按键按下,表明要显示不同的内容,此时从这个循环中跳出 break; //接下来将序列号的低32位或高32位值取出放入disbuffer_rom存储 disbuffer_rom[0]=(ID[m][k]&0x0F); disbuffer_rom[1]=((ID[m][k]&0xF0)>>4); disbuffer_rom[2]=(ID[m][k+1]&0x0F); disbuffer_rom[3]=((ID[m][k+1]&0xF0)>>4); disbuffer_rom[4]=(ID[m][k+2]&0x0F); disbuffer_rom[5]=((ID[m][k+2]&0xF0)>>4); disbuffer_rom[6]=(ID[m][k+3]&0x0F); disbuffer_rom[7]=((ID[m][k+3]&0xF0)>>4); for(q=0;q<250;q++) //显示序列号的低32位或高32位 { if(z==1) //如果z为1,有按键按下,表明要显示不同的内容,此时从这个显示循环中跳出 break; for (i=0;i<8;i++) { if(z==1) break; t=chocode[i]; //取当前的位选码 P2=t; //送出位选码 p=disbuffer_rom[i]; //取当前显示的字符 t=codevalue[p]; //查得显示字符的字段码 P0=t; //送出字段码 delay(40); } } P2=0xFF; //显示完一轮,灯灭 for(q=0;q<500;q++) { if(z==1) //如果z为1,有按键按下,表明要显示不同的内容,此时从这个显示循环中跳出 break; delay(100); } } } void display_error(void) //定义显示总线上没有DS18B20时的函数 { uchar i,p,t; uint q; uchar disbuffer_temp[8]={0,2,8,1,16,0x0f,0x0f,0}; //显示内容为"OFF 1820" uchar codevalue[]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xf8,0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e,0xff,0xbf}; //共阳极的字段码 uchar chocode[]={0xf7,0xfb,0xfd,0xfe,0x7f,0xbf,0xdf,0xef}; //位选码表 for(q=0;q<250;q++) { for (i=0;i<8;i++) { t=chocode[i]; //取当前的位选码 P2=t; //送出位选码 p=disbuffer_temp[i]; //取当前显示的字符 t=codevalue[p]; //查得显示字符的字段码 P0=t; //送出字段码 delay(40); } } } uchar search_rom(void) //遍历搜索单线上所连的所有18b20的序列号 { uchar k,l=0,chongtuwei,m,n,a; uchar _00web[MAXNUM]={0}; do { DS_init(); //复位单总线上的所有DS18B20 write_byte(0xf0); //单片机发布搜索命令 for(m=0;m<8;m++) { uchar s=0; //s用来记录本次循环得到的1个字节(8位)序列号 for(n=0;n<8;n++) { k=read_2bit(); //读第m*8+n位的原码和反码,保存在k中 k=k&0x03; //屏蔽掉k中其它位的干扰,为下一步判断作准备 s>>=1; //s右移一位,即把上一次循环得到的位值右移一位, //这样执行完一次n为变量的循环,便可得到一个字节的ROM号 if(k==0x01) //k为01,表明读到的数据为0,即所有器件在这一位都为0,所以向总线上写0 //同时对s的值不进行操作,即这位的序列号记为0 { write_bit (0); } else if(k==0x02)//k为02,表明读到的数据为1,即所有器件在这一位都为1,所以向总线上写1 { s=s|0x80; //记录下此位的值,即s的最高位置1 write_bit (1); } else if(k==0x00) { chongtuwei=m*8+n+1; //记录下这个冲突位发生的位置;之所以加1是为了让_00web数组中的第一位保持0不变, //便于判断搜索循环是否结束; if(chongtuwei>_00web[l])//如果冲突位比标志00位的位高,即发现了新的冲突位,那么这位写0 { write_bit (0); _00web[++l]=chongtuwei; //依次记录位比冲突标志位高的冲突位在_00web数组中 } else if(chongtuwei<_00web[l]) //如果冲突位比标志00位的位低,那么把ID中这位所在的字节右移n位, { //从而得到这位先前已经写过的值,如果为0,说明这位先前写的是0,那么继续写0, // 如果这位先前写的是1,那么继续写1 a=(ID[num-1][m]>>n)&0x01; s=s|(a<<7); //记录下此位的值 write_bit(a); } else if(chongtuwei==_00web[l])//如果冲突位就是标志00位,那么s的最高位置1,即这位记为1,同时向总线上写1; //之所以不写0,是因为前面已经写过0,再写0,就得不到遍历的效果. { s=s|0x80; write_bit (1); l=l-1; //改变标志00位的位置,即向前推一个00位,并且是往低位方向推 } } else if(k==0x03) //k为03,表明总线上没有DS18B20,函数结束,同时返回零值 {return(0);} } ID[num][m]=s; } num++; //DS18B20的个数加1 }while((_00web[l]!=0)&&(num<MAXNUM));//如果冲突位记录数组已经前推到0值或是DS18B20的数目已经超过最大允许数目, //就退出循环 return(1); //搜索完毕,返回1值 } void Read_Temperature_rom(void) //读取温度函数 { uchar i; DS_init(); write_byte(0x55); //匹配ROM for(i=0;i<8;i++) //发出64位ROM编码 write_byte(RomID_temp[i]); write_byte(0x44); //开始转换 DS_init(); write_byte(0x55); //匹配ROM for(i=0;i<8;i++) //发出64位ROM编码 write_byte(RomID_temp[i]); write_byte(0xBE); //发读温度命令 temp.c[1]=read_byte(); //读低字节,之所以c[1]中放低字节,是因为C51采用的是大端格式,具体详见共用体定义的说明 temp.c[0]=read_byte(); //读高字节,之所以c[0]中放低字节,是因为C51采用的是大端格式,具体详见共用体定义的说明 } void Temperature_cov(void) //温度转换 { if (temp.c[0]>0xf8) {flag=1;temp.x=~temp.x+1;} //如果为负,则符号标志置1,计算温度值,注意c[0]中放的是读取温度值的高字节 //还要注意读出的数值一共是12位 cc=temp.x/16; //计算出温度值的整数部分,这个语句相当于数值乘0.0625再取整数部分 xs=temp.x&0x0f; //取温度值小数部分的第一位 xs=xs*10; //这两条语句相当于乘0.625,得小数位的第一位,注意不是乘0.0625 xs=xs/16; } void display(void) //定义温度底层显示函数 { uchar codevalue[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71,0x00,0x40};//共阴极的字段码 uchar chocode[]={0xdf,0xef,0xf7,0xfb,0xfd,0xfe,};//位选码表 uchar i=0,p,t; disbuffer[4]=0x10; //让LCD的第5位全灭,即什么也不显示,注意是16进制表示 disbuffer[5]=0x0A+m; //LCD的第6位显示DS18B20序列号,以字母A,B,C,D,E,F来给示,所以加上0x0a if (flag==1)disbuffer[3]=0x11;//判断是否显示负号 else disbuffer[3]=0x10; disbuffer[0]=xs; //放入小数位显示的数值 disbuffer[1]=(cc%10);//放入个数要显示的数值 disbuffer[2]=(cc/10);//放入十位要显示的数值 for (i=0;i<6;i++) { /* t=chocode[i]; //取当前的位选码 P2=t; //送出位选码 p=disbuffer[i]; //取当前显示的字符 t=codevalue[p]; //查得显示字符的字段码 if (i==1) t=t+0x80; //个位比较特殊,因为有小数点,所以要加上0x80 P0=t; //送出字段码 delay(40); */ t=codevalue[disbuffer[i]]; if (i==1) t=t+0x80; dula=1; P0=t; dula=0; P0=0xff; wela=1; P0=chocode[i]; wela=0; delay(45); } } void diplay_final(void) //定义温度上层显示函数 { uint q,r; z=0; for(q=0;q<8;q++) //将要显示DS18B20的温度值的序列号放入数组RomID_temp中 { RomID_temp[q]=ID[m][q]; } P0=0xFF; if(a==0) //如果按键0的标志变量a=0,即进行闪烁显示 { for(q=0;q<3;q++) //闪烁空隙仍在读取温度 { if(z==1) //如果z为1,有按键按下,表明要显示不同的内容,此时从这个循环中跳出 break; flag=0; Read_Temperature_rom(); // 读取双字节温度 Temperature_cov(); //温度转换 for(r=0;r<15;r++) delay(1000); } } for(q=0;q<5;q++) //读取温度并显示 { if(z==1) //如果z为1,有按键按下,表明要显示不同的内容,此时从这个显示循环中跳出 break; flag=0; Read_Temperature_rom(); // 读取双字节温度 Temperature_cov(); //温度转换 for(r=0;r<100;r++) display(); //显示温度 } } void key_0() interrupt 0 //外部中断0的中断处理函数 { delay(1200); //延时消抖 if(Key0==0) { if(a==0) //a只可能取0值或1值 a=1; else a=0; b=0; //同时一旦0键按下就将b归零 z=1; //将有按键按下标志位z置位 } } void key_1() interrupt 2 //外部中断1的中断处理函数 { delay(1200); //延时消抖 if(Key1==0) { if(a==0) //只有a等于0时才让b的值在0或1之间转变 { if(b==0) b=1; else b=0; } z=1; //将有按键按下标志位z置位 if(a==1) //当a=1时进入固定显示一个DS18B20的温度值的状态 { m++; //按一次1键m值加1,即在不同的DS18B20切换显示 if(m>=num) //当然,m的值不能超过或等于总线上挂接的DS18B20的数目 m=0; } } } void main() //主函数 { uchar p,i; delay(10); p=search_rom(); //执行搜索DS18B20算法,同时返回搜索情况,p值为0,表明总线上没有DS18B20,p值为1表明搜索到DS18B20; if(p==0) //如果总线上没有DS18B20,显示"OFF 1820" while(1) display_error(); EA=1; //开中断 EX0=1; //允许外部中断0 IT0=1; //外部中断0为边沿触发方式 EX1=1; //允许外部中断1 IT1=1; //外部中断1为边沿触发方式 while(1) { if(a==0&&b==0) //按键0和1的状态变量值都为0,此时轮换显示各DS18B20的温度值 { diplay_final(); //显示DS18B20的温度值 if(m<num-1) //为了实现轮换显示,要改变m的值 m++; else m=0; if(z==1) //如果有按键按下,将m清零 { m=0; } } else if(a==0&&b==1) //按键0和按键1的状态变量值分别为0和1,此时轮换显示各DS18B20的序列号 { display_ROMID(); //显示DS18B20的序列号 if(m<num-1) //为了实现轮换显示,要改变m的值 m++; else m=0; if(z==1) //如果有按键按下,将m清零 { m=0; } } else //按键0的状态变量值为1,此时固定显示各DS18B20的温度值,按键1用于不同DS18B20间的切换 { diplay_final(); //显示DS18B20的温度值 } } }由于最近比较忙,所有过多的芯片简介就没介绍,希望大家理解。