求文档: 单片机设计——DS1820数字温度计程序与论文 交作业啊

2025-03-03 08:25:23
推荐回答(3个)
回答1:

#include
#include
#define uchar unsigned char
#define uint unsigned int
sbit ds=P3^0;
uchar flag ;
uint temp; //参数temp一定要声明为 int 型
uchar table[]={
0xc0,0xf9,0xa4,0xb0,
0x99,0x92,0x82,0xf8,
0x80,0x90,0x88,0x83,
0xc6,0xa1,0x86,0x8e};

/*延时函数*/
void TempDelay (uchar us)
{
while(us--);
}

void delay(uint count) //延时子函数
{
uint i;
while(count)
{
i=200;
while(i>0)
i--;
count--;
}
}

/*串口初始化,波特率9600,方式1 */

/*数码管的显示 */
void display()
{ uchar bai,shi,ge;
bai=temp/100;
shi=temp%100/10;
ge=temp%100%10;

P2=0xbf;
P0=table[bai]; //显示百位
delay(1); //从0到1,有个上升沿,解除锁存,显示相应段
P0=0XFF;

P2=0xdf;
P0=table[shi]&0X7F; //显示十位
delay(1);
P0=0XFF;

P2=0xef;
P0=table[ge]; //显示个位
delay(1);
P0=0xff;
}
/*****************************************
时序:初始化时序、读时序、写时序。
所有时序都是将主机(单片机)作为主设备,单总
线器件作为从设备。而每一次命令和数据的传输
都是从主机主动启动写时序开始,如果要求单总
线器件回送数据,在进行写命令后,主机需启动
读时序完成数据接收。数据和命令的传输都是低
位在先。
初始化时序:复位脉冲 存在脉冲
读;1 或 0时序
写;1 或 0时序
只有存在脉冲信号是从18b20(从机)发出的,其
它信号都是由主机发出的。
存在脉冲:让主机(总线)知道从机(18b20)已
经做好了准备。
******************************************/

/*--------------------------------------------------------------------------------------------------------------------
初始化:检测总线控制器发出的复位脉冲
和ds18b20的任何通讯都要从初始化开始

初始化序列包括一个由总线控制器发出的复位脉冲
和跟在其后由从机发出的存在脉冲。

初始化:复位脉冲+存在脉冲

具体操作:
总线控制器发出(TX)一个复位脉冲 (一个最少保持480μs 的低电平信号),然后释放总线,
进入接收状态(RX)。单线总线由5K 上拉电阻拉到高电平。探测到I/O 引脚上的上升沿后
DS1820 等待15~60μs,然后发出存在脉冲(一个60~240μs 的低电平信号)。

具体看"倒塌 18b20"文档里的 " 单线复位脉冲时序和1-wire presence detect "的时序图
-------------------------------------------------------------------------------------------------------------------*/
void ds_reset(void)
{
ds=1;
_nop_(); //1us
ds=0;
TempDelay(80); //当总线停留在低电平超过480us,总线上所以器件都将被复位,这里//延时约530us总线停留在低电平超过480μs,总线上的所有器件都
//将被复位。
_nop_();
ds=1; //产生复位脉冲后,微处理器释放总线,让总线处于空闲状态,原因查//18b20中文资料

TempDelay(5); //释放总线后,以便从机18b20通过拉低总线来指示其是否在线,
//存在检测高电平时间:15~60us, 所以延时44us,进行 1-wire presence //detect(单线存在检测)
_nop_();
_nop_();
_nop_();
if(ds==0)
flag=1; //detect 18b20 success
else
flag=0; //detect 18b20 fail
TempDelay(20); //存在检测低电平时间:60~240us,所以延时约140us
_nop_();
_nop_();
ds=1; //再次拉高总线,让总线处于空闲状态
/**/
}

/*----------------------------------------
读/写时间隙:
DS1820 的数据读写是通过时间隙处理
位和命令字来确认信息交换。
------------------------------------------*/
bit ds_read_bit(void) //读一位
{
bit dat;
ds=0; //单片机(微处理器)将总线拉低
_nop_(); //读时隙起始于微处理器将总线拉低至少1us
ds=1; //拉低总线后接着释放总线,让从机18b20能够接管总线,输出有效数据
_nop_();
_nop_(); //小延时一下,读取18b20上的数据 ,因为从ds18b20上输出的数据
//在读"时间隙"下降沿出现15us内有效
dat=ds; //主机读从机18b20输出的数据,这些数据在读时隙的下降沿出现//15us内有效
TempDelay(10); //所有读"时间隙"必须60~120us,这里77us
return(dat); //返回有效数据
}

uchar ds_read_byte(void ) //读一字节
{

uchar value,i,j;
value=0; //一定别忘了给初值
for(i=0;i<8;i++)
{
j=ds_read_bit();
value=(j<<7)|(value>>1); //这一步的说明在一个word文档里面
}
return(value); //返回一个字节的数据
}

void ds_write_byte(uchar dat) //写一个字节
{
uchar i;
bit onebit; //一定不要忘了,onebit是一位
for(i=1;i<=8;i++)
{
onebit=dat&0x01;
dat=dat>>1;
if(onebit) //写 1
{
ds=0;
_nop_();
_nop_(); //看时序图,至少延时1us,才产生写"时间隙"
ds=1; //写时间隙开始后的15μs内允许数据线拉到高电平
TempDelay(5); //所有写时间隙必须最少持续60us
}
else //写 0
{
ds=0;
TempDelay(8); //主机要生成一个写0 时间隙,必须把数据线拉到低电平并保持至少60μs,这里64us
ds=1;
_nop_();
_nop_();
}
}
}

/*****************************************
主机(单片机)控制18B20完成温度转换要经过三个步骤:
每一次读写之前都要18B20进行复位操作,复位成功后发送
一条ROM指令,最后发送RAM指令,这样才能对DS18b20进行
预定的操作。
复位要求主CPU将数据线下拉500us,然后释放,当ds18B20
受到信号后等待16~60us,后发出60~240us的存在低脉冲,
主CPU收到此信号表示复位成功
******************************************/

/*----------------------------------------
进行温度转换:
先初始化
然后跳过ROM:跳过64位ROM地址,直接向ds18B20发温度转换命令,适合单片工作
发送温度转换命令
------------------------------------------*/

void ds18b20()
{
ds_reset();
delay(1); //约2ms
ds_write_byte(0xcc);
ds_write_byte(0x44);
}

/*----------------------------------------
获得温度:
------------------------------------------*/
uint get_temperature()
{
float wendu;
uchar a,b;
ds_reset();
delay(1); //约2ms
ds_write_byte(0xcc);
ds_write_byte(0xbe);
a=ds_read_byte();
b=ds_read_byte();
temp=b;
temp<<=8;
temp=temp|a;
wendu=temp*0.0625;
temp=wendu*10+0.5;
return temp;
}
/*----------------------------------------
读ROM
------------------------------------------*/
/*
void ds_read_rom() //这里没有用到
{
uchar a,b;
ds_reset();
delay(30);
ds_write_byte(0x33);
a=ds_read_byte();
b=ds_read_byte();
}
*/
void main()
{
uint a;

while(1)
{
ds18b20(); //12位转换时间最大为750ms
for(a=10;a>0;a--)
{get_temperature();
display( );
}
}
}

连接很简单,就在最小系统的p0口接数码管,在p3.7口接上DS18B20电路不封就可以

回答2:

; 程序功能:数字温度计

       A_BIT   EQU  20H     ;存放个位数变量

       B_BIT   EQU  21H     ;存放十位数变量

       FLAG    EQU  38H     ;DS18B20是否存在标志

       DQ      EQU  P2.5    ;DQ引脚由P2.5控制  

  MAIN:                     ;主程序标号

       ACALL  RE_TEMP      ;对传感器设置及读取 

       ACALL   TURN     ;转化温度子程序        

       ACALL DISPLAY        ;显示子程序

       JMP MAIN             ;循环                   

 

  RE_TEMP:         ;对DS18B20初始化及读取

       SETB   DQ           

       ACALL  RESET_1820    ;调用复位子程序

       JB   FLAG, ST        ;判断DS1820是否存在?

       RET                                           

 

  ST:                       ; DS18B20存在

       MOV   A,#0CCH        ;跳过ROM匹配 

       ACALL WRITE_1820     ;写入数据

       MOV  A,#44H          ;发出温度转换命令

       ACALL WRITE_1820     ;写入数据 

       ACALL  RESET_1820    ;准备读温度前先复位

       MOV  A,#0CCH         ;跳过ROM匹配

       ACALL WRITE_1820     ;写入数据

       MOV  A,#0BEH         ;发出读温度命令

       ACALL WRITE_1820     ;写入数据

       ACALL READ_1820      ;读出温度数据         

       RET                                         

  RESET_1820:  ;复位(有具体的时序要求)                       

       SETB  DQ       

       NOP

       CLR   DQ       

 ;――――――――――――

 ;主机发出延时537微秒的复位低脉冲

       MOV  R1,#3

   DLY: MOV  R0,#107

       DJNZ R0,$

       DJNZ R1,DLY

 ;――――――――――――

 ;然后拉高数据线

       SETB  DQ        

       NOP

       NOP

       NOP

 ;――――――――――――  

       MOV  R0,#25H        

   T2: JNB   DQ ,T3       ;等待DS18B20回应

       DJNZ  R0, T2

       JMP  T4          

 ;――――――――――――

   T3: SETB  FLAG      ;置标志位,表示DS1820存在

       JMP  T5

 ;――――――――――――

   T4: CLR   FLAG     ;清标志位,表示DS1820不存在     

       JMP  T7

  ;――――――――――――

   T5: MOV  R0,#117

   T6: DJNZ R0,T6           ;时序要求延时一段时间

 ;――――――――――――

   T7: SETB   DQ        

       RET                                           

 ;――――――――――――

  WRITE_1820: ;写入DS18B20(有具体的时序要求)

       MOV  R2,#8           ;一共8位数据

       CLR  C

    WR1:

       CLR  DQ            ;总线低位,开始写入  

       MOV  R3,#7

       DJNZ R3,$            ;保持16微秒以上

       RRC  A  ;把字节DATA分成8个BIT环移给C

       MOV  DQ,  C        ;写入一个BIT

       MOV  R3,#23

       DJNZ R3,$            ;等待

       SETB  DQ           ;重新释放总线

       NOP

       DJNZ R2,WR1          ;写入下一个BIT

       SETB  DQ          

       RET                                            

   READ_1820: ;将温度高位和低位从DS18B20中读出

       MOV  R4,#2           ; 读出两个字节的数据

       MOV  R1,#29H     ;低位存入29H,高位存入28h

   RE0:

       MOV R2,#8           ;数据一共有8位

   RE1:                  

       CLR C

       SETB  DQ          

       NOP

       NOP                                 

       CLR   DQ          ;读前总线保持为低               

       NOP

       NOP

       NOP

       SETB  DQ          ;开始读总线释放                 

       MOV   R3,#9

   RE2:   

       DJNZ  R3,RE2        ;延时18微妙

      MOV C, DQ         ;从总线读到一个BIT                  

      MOV R3,#23

  RE3:                    ;

      DJNZ R3,RE3         ;等待100秒

      RRC  A              ;把读得的位价值环移给A

      DJNZ  R2,RE1        ;读下一个BIT

      MOV  @R1,A

      DEC  R1

      DJNZ R4,RE0

      RET                                         

  TURN: 

      MOV  A,29H

      MOV  C,40H          ;将28中的最低位移入C

      RRC  A

      MOV  C,41H

      RRC  A

      MOV  C,42H

      RRC  A

      MOV  C,43H

      RRC  A

      MOV  29H,A

      RET                                           

  DISPLAY:   ;将29H中的十六进制数成10进制

      MOV  A,29H         

      MOV  B,#10             ;10进制/10=10进制

      DIV  AB

      MOV  B_BIT, A        ;十位在A

      MOV  A_BIT, B        ;个位在B

      MOV  DPTR,#TABLE     ;指定查表启始地址

      MOV  R0,#4

  DP1:

      MOV R1, #250          ;显示1000次

  LOOP:

      MOV A,A_BIT          ;取个位数

      MOVC A,@A+DPTR       ;查个位数的7段代码

      MOV  P0,A            ;送出个位的7段代码

      CLR  P2.3            ;开个位显示

      ACALL DELAY          ;

      SETB  P2.3

      MOV A,B_BIT          ;取十位数

      MOVC  A,@A+DPTR      ;查十位数的7段代码

      MOV  P0,A            ;送出个十位的7段代码

      CLR  P2.2            ;开十位显示

      ACALL  DELAY         ;显示1MS

      SETB  P2.2

      DJNZ  R1,LOOP        ;250次没完循环

      DJNZ  R0,DP1         ;4个250次没完循环

      RET                                         

   DELAY:  ;1ms延时

      MOV  R7, #80

      DJNZ R7,$

      RET                                           

  TABLE:       ;数码管共阳极0~9代码

      DB 0C0H,0F9H,0A4H,0B0H,99H

      DB 92H,82H,0F8H,80H,90H                       

      END

回答3:

程序与论文我这有,怎么发给你啊?我可以帮助你!!