1、效果图
#include
sbit DQ = P2^0; //定义总线的I/O管脚
sbit dm = P2^2; //段码
sbit wm = P2^3; //位码
unsigned char hc[8]={0x3f,0x3f,0x3f}; //显示缓存
unsigned char DM[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};// 显示段码值0~F
unsigned char WM[]={0xdf,0xbf,0x7f};//分别对应相应的数码管点亮,即位码
void SendByte(unsigned char dat);
void Delay(unsigned char j);
void sm() //数码管扫描函数1(95 + 45= 150 us)
{
unsigned char k=3;
while (k--)
{
P1 = 0; //消影
dm = 1;
dm = 0;
P1 = WM[k]; //写入位码
wm = 1;
wm = 0;
P1 = hc[k]; //写入段码
dm = 1;
dm = 0;
Delay(1);
}
}
void sm2(unsigned char k) ////数码管扫描函数2(19+15+4 大概 42us)
{
P1 = 0; //消影
dm = 1;
dm = 0;
P1 = WM[k]; //写入位码
wm = 1;
wm = 0;
P1 = hc[k]; //写入段码
dm = 1;
dm = 0;
Delay(1);
}
void Delay6us() //空闲7个指令周期
{
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
}
void Delay(unsigned char j) //一个循环15us
{
unsigned char i;
while(j--)
{
i = 5;
while (--i);
}
}
bit d18b20_qs() //18b20 起始
{
bit dat;
DQ = 1; //DQ复位
_nop_();
DQ = 0; //拉低总线
sm(); //这里延时 150us
Delay(15); //这里延时大概 225us
sm();//这里延时 150us
DQ = 1; //拉高总线
Delay(2); //这里延时大概 30us
dat = DQ; //读取返回值(0:有18b20存在 1:是没有)
Delay(2);
return dat; //返回数值
}
[page]
void d18b20_x(unsigned char dat) //写 8 位 数 据
{
unsigned char i;
for(i=0;i<8;i++) //8位计数器
{
DQ = 0; //拉低总线
DQ = dat & 0x01; //取最低位赋值给总线
sm2(2); //延时45us
DQ = 1; //拉过总线准备写下一个数据(或者总线复位)
dat >>= 1; //数据右移一位
}
}
unsigned char d18b20_d() //读 8 位 数 据
{
unsigned char i,dat=0;
for(i=0;i<8;i++) //8位计数器
{
DQ = 0; //拉低总线
dat >>= 1; //数据右移一位
DQ = 1; //拉过总线(准备读取数据)
if(DQ) //判断是否是 1 如果是就把数据赋值给变量的高位
dat |= 0x80;
sm2(1);
}
return dat; //返回读取到数据数据
}
unsigned int wd() //读取温度函数
{
unsigned char i = 0; //低8位数据
unsigned char j = 0; //高8位数据
unsigned int k = 0; //无符号16整形用来存储读回来的 16位温度数据(j和i组合后的数据)
d18b20_qs(); //初始化
d18b20_x(0xCC); //跳过序列号的操作(因为18b20在总线上可以挂很多个,这个序列号和网卡MAC地址类似)
d18b20_x(0x44); //开启温度转换
for(;k<15;k++) //开启温度转换需要时间这里延时一下15*150us
sm();
d18b20_qs(); //初始化
d18b20_x(0xCC); //跳过序列号的操作(因为18b20在总线上可以挂很多个,这个序列号和网卡MAC地址类似)
d18b20_x(0xBE); //读取温度寄存器等(共可读9个寄存器) 前两个就是温度
i = d18b20_d(); //读取低8位
sm();
j = d18b20_d(); //读取高8位
k = j;
k <<= 8;
k = k + i;
return k; //返回读取到的16位数据
}
void zh(unsigned int i) //数码管显示缓存写入函数
{
unsigned char x,z;
x = i & 0x0f; //取出小数
i >>=4;
z = i & 0xff; //取出整数
switch(x) //小数位写人显示缓存
{
case 0: hc[2]=DM[0];break;
case 1: hc[2]=DM[1];break;
case 2: hc[2]=DM[1];break;
case 3: hc[2]=DM[2];break;
case 4: hc[2]=DM[3];break;
case 5: hc[2]=DM[3];break;
case 6: hc[2]=DM[4];break;
case 7: hc[2]=DM[4];break;
case 8: hc[2]=DM[5];break;
case 9: hc[2]=DM[6];break;
case 10: hc[2]=DM[6];break;
case 11: hc[2]=DM[7];break;
case 12: hc[2]=DM[8];break;
case 13: hc[2]=DM[8];break;
case 14: hc[2]=DM[9];break;
case 15: hc[2]=DM[9];break;
}
x = z/10; //取出十位
hc[0]=DM[x];//十位缓存写入
x = z%10; //取出个位
hc[1] = DM[x] | 0x80;//个位缓存写入
}
/******************************串口操作函数********************************
void CSH (void) //初始化串口
{
SCON = 0x50; // SCON: 模式 1, 8-bit UART, 使能接收
TMOD |= 0x20; // TMOD: timer 1, mode 2, 8-bit 重装
TH1 = 0xFD; // TH1: 重装值 9600 波特率 晶振 11.0592MHz
TR1 = 1; // TR1: timer 1 打开
EA = 1; //打开总中断
//ES = 1; //打开串口中断
}
void SendByte(unsigned char dat) //发送一个字符
{
SBUF = dat; //SBUF 串行数据缓冲器
while(!TI); //TI发送中断标志位 (当数据发送完毕后由硬件置 1 否则等待硬件置 1)
TI = 0;
}
*********************************串口操作函数**************************************/
void main()
{
unsigned char k,i;
while(1)
{
zh(wd());
for(k=0;k<200;k++)
{
sm();
for(i=0;i<5;i++)
Delay6us();
}
}
}