scan()
{
char k
for(k=0k<4k++) //4位LED扫描控制
{
discan=0x00
Disdata=dis_7[_1820display[k]] //数据显示
if (k==1){DIN=0} //小数点显示
discan=scan_con[k]//位选
_18B20_delay(100)
}
}
main()
_18B20_init()//18B20初始化
while(1)
{
EA=0//在利用18B20测试温度时,要严格遵循时序,禁止一切中断
_18B20_work(_18B20_read())//处理温度数据
EA=1//测试完毕,恢复系统中断
scan()//显示温度值
}
*/
#include "intrins.h"//_nop_()延时函数用
//*****************//
//以下是DS18B20驱动程序
//*****************//
/**************************************************
** 功能描述: DS18B20驱动程序,使用12M晶体
** DQ占用引脚资源P1^7
****************************************************/sbit DQ=P1^7 //温度输入口unsigned char data temp_data[2]={0x00,0x00}//读出温度暂放
unsigned char data _1820display[5]={0x00,0x00,0x00,0x00,0x00}//显示单元数据,共4个数据和一个运算暂用
unsigned int temp
//**************温度小数部分用查表法***********//
unsigned char code ditab[16]=
{
0x00,0x01,0x01,0x02,
0x03,0x03,0x04,0x04,
0x05,0x06,0x06,0x07,
0x08,0x08,0x09,0x09
}/*****************11us延时函数*************************/
//
void _18B20_delay(unsigned int t)
{
for (t>0t--)
}/****************DS18B20复位函数************************/
_18B20_reset(void)
{
char presence=1
while(presence)
{
while(presence)
{
DQ=1
_nop_()_nop_()//从高拉倒低
DQ=0
_18B20_delay(50) //550 us
DQ=1
_18B20_delay(6) //66 us
presence=DQ//presence=0 复位成功,继续下一步
}
_18B20_delay(45) //延时500 us
presence=~DQ
}
DQ=1 //拉高电平
}/****************DS18B20写命令函数************************/
//向1-WIRE 总线上写1个字节
void _18B20_write(unsigned char val)
{
unsigned char i
for(i=8i>0i--)
{
DQ=1
_nop_()_nop_() //从高拉倒低
DQ=0
_nop_()_nop_()_nop_()_nop_() //5 us
DQ=val&0x01 //最低位移出
_18B20_delay(6) //66 us
val=val/2//右移1位
}
DQ=1
_18B20_delay(1)
}/****************DS18B20读1字节函数************************/
//从总线上取1个字节
unsigned char _18B20read_byte(void)
{
unsigned char i
unsigned char value=0
for(i=8i>0i--)
{
DQ=1
_nop_()_nop_() //从高拉倒低
value>>=1
DQ=0
_nop_()_nop_()_nop_()_nop_()//4 us
DQ=1
_nop_()_nop_()_nop_()_nop_()//4 us
if(DQ)value|=0x80
_18B20_delay(6)//66 us
}
DQ=1
return(value)
}
_18B20_read() //读出温度函数
{
_18B20_reset() //总线复位
_18B20_delay(200)
_18B20_write(0xcc) //发命令
_18B20_write(0x44) //发转换命令
_18B20_reset()
_18B20_delay(1)
_18B20_write(0xcc) //发命令
_18B20_write(0xbe)
temp_data[0]=_18B20read_byte() //读温度值的低字节
temp_data[1]=_18B20read_byte() //读温度值的高字节
temp=temp_data[1]
temp<<=8
temp=temp|temp_data[0] // 两字节合成一个整型变量。
return temp//返回温度值
}/****************温度数据处理函数************************///二进制高字节的低半字节和低字节的高半字节组成一字节,这个
//字节的二进制转换为十进制后,就是温度值的百、十、个位值,而剩
//下的低字节的低半字节转化成十进制后,就是温度值的小数部分/********************************************************/
_18B20_work(unsigned int tem)
{
unsigned char n=0
if(tem>6348)// 温度值正负判断
{
tem=65536-tem
n=1
}// 负温度求补码,标志位置1
_1820display[4]=tem&0x0f // 取小数部分的值
_1820display[0]=ditab[_1820display[4]] // 存入小数部分显示值
_1820display[4]=tem>>4// 取中间八位,即整数部分的值
_1820display[3]=_1820display[4]/100// 取百位数据暂存
_1820display[1]=_1820display[4]%100// 取后两位数据暂存
_1820display[2]=_1820display[1]/10 // 取十位数据暂存
_1820display[1]=_1820display[1]%10
/******************数码管符号位显示判断**************************/
if(!_1820display[3])
{
_1820display[3]=0x0a //最高位为0时不显示
if(!_1820display[2])
_1820display[2]=0x0a //次高位为0时不显示
}
if(n)
_1820display[3]=0x0b //负温度时最高位显示"-"
}
/******************1602液晶符号位显示判断**************************/
if(!_1820display[3])
{
_1820display[3]=' '-'0' //最高位为0时不显示
if(!_1820display[2])
_1820display[2]=' '-'0' //次高位为0时不显示
}
if(n)
_1820display[3]='-'-'0' //负温度时最高位显示"-"
} _18B20_init()//18B20初始化
{
_18B20_reset() //开机先转换一次
_18B20_write(0xcc)//Skip ROM
_18B20_write(0x44)//发转换命令
}
我在10年做了一个差不多的,也是报警控制的,你借鉴一下吧:
软件设计:
有两个文件,DS18B20.c和DS18B20.h,将这两个文件添加到工程里即可。
DS18B20.c:
/******************************************************************
程序名称:DS18B20温度测量、报警系统
简要说明:DS18B20温度计,温度测量范围0~99.9摄氏度
可设置上限报警温度、下限报警温度
即高于上限值或者低于下限值时蜂鸣器报警
默认上限报警温度为32℃、默认下限报警温度为10℃
报警值可设置范围:最低上限报警值等于当前下限报警值
最高下限报警值等于当前上限报警值
将下限报警值调为0时为关闭下限报警功能
******************************************************************/
#include <AT89X52.h>
#include "DS18B20.h"
#define uint unsigned int
#define uchar unsigned char //宏定义
#define SET P3_1 //定义调整键
#define DEC P3_2 //定义减少键
#define ADD P3_3 //定义增加键
#define BEEP P3_7 //定义蜂鸣器
#define JDQ P3_5
bit shanshuo_st //闪烁间隔标志
bit beep_st //蜂鸣器间隔标志
sbit DIAN = P2^7 //小数点
uchar x=0 //计数器
signed char m //温度值全局变量
uchar n //温度值全局变量
uchar set_st=0 //状态标志
signed char shangxian=70 //上限报警温度,默认值为70
signed char xiaxian=0 //下限报警温度,默认值为0
uchar code LEDData[]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0xff}
/*****延时子程序*****/
void Delay(uint num)
{
while( --num )
}
void shortdelay()(void) //误差 0us
{
unsigned char a,b,c
for(c=165c>0c--)
for(b=100b>0b--)
for(a=150a>0a--)
_nop_ //if Keil,require use intrins.h
_nop_ //if Keil,require use intrins.h
}
/*****初始化定时器0*****/
void InitTimer(void)
{
TMOD=0x1
TH0=0x3c
TL0=0xb0 //50ms(晶振12M)
}
/*****定时器0中断服务程序*****/
void timer0(void) interrupt 1
{
TH0=0x3c
TL0=0xb0
x++
}
/*****外部中断0服务程序*****/
void int0(void) interrupt 0
{
EX0=0 //关外部中断0
if(DEC==0&&set_st==1)
{
shangxian--
if(shangxian<xiaxian)shangxian=xiaxian
}
else if(DEC==0&&set_st==2)
{
xiaxian--
if(xiaxian<0)xiaxian=0
}
}
/*****外部中断1服务程序*****/
void int1(void) interrupt 2
{
EX1=0 //关外部中断1
if(ADD==0&&set_st==1)
{
shangxian++
if(shangxian>99)shangxian=99
}
else if(ADD==0&&set_st==2)
{
xiaxian++
if(xiaxian>shangxian)xiaxian=shangxian
}
}
/*****读取温度*****/
void check_wendu(void)
{
uint a,b,c
c=ReadTemperature()-5 //获取温度值并减去DS18B20的温漂误差
a=c/100 //计算得到十位数字
b=c/10-a*10 //计算得到个位数字
m=c/10 //计算得到整数位
n=c-a*100-b*10 //计算得到小数位
if(m<0){m=0n=0} //设置温度显示上限
if(m>99){m=99n=9} //设置温度显示上限
}
/*****显示开机初始化等待画面*****/
Disp_init()
{
P2 = 0xbf //显示-
P1 = 0xf7
Delay(200)
P1 = 0xfb
Delay(200)
P1 = 0xfd
Delay(200)
P1 = 0xfe
Delay(200)
P1 = 0xff //关闭显示
}
/*****显示温度子程序*****/
Disp_Temperature() //显示温度
{
P2 =0xc6 //显示C
P1 = 0xf7
Delay(300)
P2 =LEDData[n] //显示个位
P1 = 0xfb
Delay(300)
P2 =LEDData[m%10] //显示十位
DIAN = 0 //显示小数点
P1 = 0xfd
Delay(300)
P2 =LEDData[m/10] //显示百位
P1 = 0xfe
Delay(300)
P1 = 0xff //关闭显示
}
/*****显示报警温度子程序*****/
Disp_alarm(uchar baojing)
{
P2 =0xc6 //显示C
P1 = 0xf7
Delay(200)
P2 =LEDData[baojing%10] //显示十位
P1 = 0xfb
Delay(200)
P2 =LEDData[baojing/10] //显示百位
P1 = 0xfd
Delay(200)
if(set_st==1)P2 =0x89
else if(set_st==2)P2 =0xc7 //上限H、下限L标示
P1 = 0xfe
Delay(200)
P1 = 0xff //关闭显示
}
/*****报警子程序*****/
void Alarm()
{
if(x>=10){beep_st=~beep_stx=0}
if((m>=shangxian&&beep_st==1)||(m<xiaxian&&beep_st==1))BEEP=0
else BEEP=1
if((m>=shangxian)||(m<xiaxian))
{shortdelay()()
JDQ=0}
else JDQ=1
}
/*****主函数*****/
void main(void)
{
uint z
InitTimer() //初始化定时器
EA=1 //全局中断开关
TR0=1
ET0=1 //开启定时器0
IT0=1
IT1=1
check_wendu()
check_wendu()
for(z=0z<300z++)
{
Disp_init()
}
while(1)
{
if(SET==0)
{
Delay(2000)
do{}while(SET==0)
set_st++x=0shanshuo_st=1
if(set_st>2)set_st=0
}
if(set_st==0)
{
EX0=0 //关闭外部中断0
EX1=0 //关闭外部中断1
check_wendu()
Disp_Temperature()
Alarm() //报警检测
}
else if(set_st==1)
{
BEEP=1 //关闭蜂鸣器
EX0=1 //开启外部中断0
EX1=1 //开启外部中断1
if(x>=10){shanshuo_st=~shanshuo_stx=0}
if(shanshuo_st) {Disp_alarm(shangxian)}
}
else if(set_st==2)
{
BEEP=1 //关闭蜂鸣器
EX0=1 //开启外部中断0
EX1=1 //开启外部中断1
if(x>=10){shanshuo_st=~shanshuo_stx=0}
if(shanshuo_st) {Disp_alarm(xiaxian)}
}
}
}
/*****END*****/
DS18B20.h:
#include <AT89X52.h>
#define DQ P3_6 //定义DS18B20总线I/O
/*****延时子程序*****/
void Delay_DS18B20(int num)
{
while(num--)
}
/*****初始化DS18B20*****/
void Init_DS18B20(void)
{
unsigned char x=0
DQ = 1 //DQ复位
Delay_DS18B20(8) //稍做延时
DQ = 0 //单片机将DQ拉低
Delay_DS18B20(80) //精确延时,大于480us
DQ = 1 //拉高总线
Delay_DS18B20(14)
x = DQ //稍做延时后,如果x=0则初始化成功,x=1则初始化失败
Delay_DS18B20(20)
}
/*****读一个字节*****/
unsigned char ReadOneChar(void)
{
unsigned char i=0
unsigned char dat = 0
for (i=8i>0i--)
{
DQ = 0 // 给脉冲信号
dat>>=1
DQ = 1 // 给脉冲信号
if(DQ)
dat|=0x80
Delay_DS18B20(4)
}
return(dat)
}
/*****写一个字节*****/
void WriteOneChar(unsigned char dat)
{
unsigned char i=0
for (i=8 i>0 i--)
{
DQ = 0
DQ = dat&0x01
Delay_DS18B20(5)
DQ = 1
dat>>=1
}
}
/*****读取温度*****/
unsigned int ReadTemperature(void)
{
unsigned char a=0
unsigned char b=0
unsigned int t=0
float tt=0
Init_DS18B20()
WriteOneChar(0xCC) //跳过读序号列号的 *** 作
WriteOneChar(0x44) //启动温度转换
Init_DS18B20()
WriteOneChar(0xCC) //跳过读序号列号的 *** 作
WriteOneChar(0xBE) //读取温度寄存器
a=ReadOneChar() //读低8位
b=ReadOneChar() //读高8位
t=b
t<<=8
t=t|a
tt=t*0.0625
t= tt*10+0.5 //放大10倍输出并四舍五入
return(t)
}
/*****END*****/
其中控制部分我用的是5V继电器,可以直接控制你的电机了。
两个电路图都差不多的,只不过我的多了几个调整按键,报警温度可以调的。我的这个程序你完全可以用到你的电路里的
#include <intrins.h>#define uchar unsigned char
#define uint unsigned int
#define out P0
#define INT8U unsigned char //宏定义
#define INT16U unsigned int
sbit smg1=P2^0
sbit smg2=P2^1
sbit smg3=P2^2
sbit smg4=P2^3
sbit Beep=P1^5//蜂鸣器引脚定义
sbit led=P1^6
sbit led1=P1^7//设置灯光报警键
sbit DQ=P2^4//ds18b20端口
void init_ds18b20(void)//ds18b20初始化子程序
void delay(uchar)//ds18b20工作延时子程序
uchar readbyte(void)//向ds18b20读一个字节数据
/*******************************************************************************/
void writebyte(uchar)//向ds18b20写一个字节数据
uint retemp()//计数变量
uchar key
uchar a,b,c,d//计数变量
uchar x[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90}
uint retemp()
{
uint a,b,t
init_ds18b20() //初始化ds18b20
writebyte(0xcc)// 跳过读序列号的 *** 作
writebyte(0x44)// 启动温度转换
init_ds18b20()
writebyte(0xcc)//跳过读序号列号的 *** 作
writebyte(0xbe)//读取温度寄存器等(共可读9个寄存器) 前两个就是温度
a=readbyte()//读出温度低位LSB
b=readbyte()//读出温度高位MSB
t=b //将温度高八位送t
t<<=8 //乘以256移到高八位
t=t|a //高低八位组合成温度值
if(t<0x8000) //如果温度为正计算正温度值
{
key=0
t=t*0.625
}
else //否则温度为负,取反
{
key=1
t=(~t+1)*0.625
}
return(t) //返回温度值
}
void main()
{
uint i,t
EA = 1//开总中断
TMOD = 0x01//定时器0工作方式1
TR0=1
delay(100)
while(1)
{
t=retemp() 读温度值
a=x[t/1000] //温度千位数
b=x[t/100%10] //温度百位数
c=x[t/10%10]-0x80 //温度十位数
d=x[t%10] //温度个位数
if(key==1) //如果key=1
a=0xbf //a为“负号"
if((key==0)&&(t>320)) //如果key=0 且t大于320
{
led1=0 //点亮led1
ET0=1 //开启定时器0中断
}
else if(t<290) //如果温度小于290
{
led=0 //点亮led
ET0=1 //开启定时器0中断
}
else //否则
{
led1=1 //关闭led1
led=1 //关闭led
ET0=0 //关闭定时器0中断
}
for(i=0i<50i++) //循环50次
{smg1=1P0=adelay(100)smg1=0 //显示千位
smg2=1P0=bdelay(100)smg2=0 //显示百位
smg3=1P0=cdelay(100)smg3=0 //显示十位
smg4=1P0=ddelay(100)smg4=0 //显示个位
}
}
}
/*ds18b20工作延时子程序*/
void delay (uchar i)
{
do
{_nop_()
_nop_()
_nop_()
i--
}
while(i)}
/*ds18b20初始化子程序*/
void init_ds18b20()
{
uchar x=0
DQ=0//单片机将DQ拉低
delay (120)
DQ=1//拉高总线
delay(16)
delay(80)
}
/*读一个字节*/
uchar readbyte ()
{uchar i=0,date=0
for(i=8i>0i--)
{
DQ=0// 给脉冲信号
delay(1)
DQ=1// 给脉冲信号
date>>=1
if(DQ)date|=0x80
delay(11)
}
return(date)
}
/*写一个字节*/
void writebyte(uchar dat)
{uchar i=0
for(i=8i>0i--) //写8位数
{
DQ=0
DQ=dat&0x01 //写dat的D0位
delay(12)
DQ=1
dat>>=1
delay(5)
}
}
/**************************************************
*函数名:中断函数
*描 述:产生矩形脉冲使蜂鸣器发声
**************************************************/
void BeepTimer0(void) interrupt 1
{
Beep = ~Beep
TH0 = 65335 / 256//定时器赋初值
TL0 = 65335 % 256
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)