本项目是51单片机的学习记录,通过串口发送字符串55,返回温度值。
开门见山,直接写。
只作为记录,实际运用还需要更改。
目录
一、代码内容
main.c
首先是main主函数,因为只使用了定时器,所以while主循环体里面没有内容。
main.h
#ifndef _MAIN_H
#define _MAIN_H
#include "regx52.h"
#define uchar unsigned char
#define uint unsigned int
typedef unsigned char u8;
typedef unsigned char u16;
#endif
main头文件里面仅仅针对文件类型作出了定义。
ds18b20.c
#include "ds18b20.h"
#include "intrins.h"
//温度采集模块
/******************************************************************************/
//发送数据
void tmpDelay(int num)//延时函数
{
while(num--) ;
}
void Send(unsigned char dat)
{
SBUF=dat;
while(TI==0) //等待发送完毕
;
TI=0;
//tmpDelay(1000);
}
//温度采集模块
/******************************************************************************/
void Init_DS18B20()//初始化ds1820
{
unsigned char x=0;
DQ = 1; //DQ复位
tmpDelay(8); //稍做延时
DQ = 0; //单片机将DQ拉低
tmpDelay(80); //精确延时 大于 480us
DQ = 1; //拉高总线
tmpDelay(14);
x=DQ; //稍做延时后 如果x=0则初始化成功 x=1则初始化失败
tmpDelay(20);
}
unsigned char ReadOneChar()//读一个字节
{
unsigned char i=0;
unsigned char dat = 0;
for (i=8;i>0;i--)
{
DQ = 0; // 给脉冲信号
dat>>=1;
DQ = 1; // 给脉冲信号
if(DQ)
dat|=0x80;
tmpDelay(4);
}
return(dat);
}
void WriteOneChar(unsigned char dat)//写一个字节
{
unsigned char i=0;
for (i=8; i>0; i--)
{
DQ = 0;
DQ = dat&0x01;
tmpDelay(5);
DQ = 1;
dat>>=1;
}
}
unsigned int Readtemp()//读取温度
{
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; //得到真实十进制温度值,因为DS18B20可以精确到0.0625度,所以读回数据的最低位代表的是0.0625度
t= tt*10+0.5; //放大十倍,这样做的目的将小数点后第一位也转换为可显示数字,同时进行一个四舍五入操作。
return(t);
}
void display()
{
unsigned int num ; //定义的时候用uchar宏定义就会出错
unsigned char shi,ge,xiaoshu; //这里的num,shi,ge,xiaoshu 必须用unsigned int无符号整数来表示,用unshigned char 字符型则显示错误
num=Readtemp();
shi=num/100+48;
ge=num/10%10+48;
xiaoshu=num%10+48;
Send(shi);
Send(ge);
Send('.');
Send(xiaoshu);
}
ds18b20,本文章的重要内容,可以通过串口输出温度信息。
ds18b20.h
#ifndef _DS18B20_H
#define _DS18B20_H
#include "regx52.h"
#include "main.h"
//管脚定义
sbit DQ = P3^7;//定义ds18B20总线IO
//函数声明
void tmpDelay(int num);
void Init_DS18B20();
unsigned int Readtemp();
void Send(unsigned char dat);
void display();
unsigned char ReadOneChar();
void WriteOneChar(unsigned char dat);
#endif
头文件可以修改定义,以确定ds18b20的引脚。这里使用的是普中的C51单片机,所以这就是默认的P3^7引脚。
delay.c
这是delay延时函数。
uart.c
#include "uart.h"
#include "ds18b20.h"
#include "delay.h"
sbit LED1 = P2^0;
char UART_RecBuf[6]={0};
char rec_index=0;
void UART_Init(u8 baud)
{
PCON |= 0x80; //使能波特率倍速位SMOD
SCON = 0x50; //8位数据,可变波特率
TMOD |= 0x20; //设置定时器模式
TH1=baud; // 设置定时器1的高字节,用于设置波特率
TL1=baud; // 设置定时器1的低字节,用于设置波特率
ET1 = 0; //禁止定时器中断
TR1 = 1; //定时器1开始计时
EA = 1;
ES = 1;
}
void tim_Init(){
TMOD |= 0x01;
TL0 = 0xB7;
TH0 = 0xFF;
ET0 = 1;
EA = 1;
TR0 = 1;
}
void uart() interrupt 4 //串口通信中断函数
{
if(RI)
{
u16 rec_data = 0;
rec_data=SBUF; //存储(取出)接收到的数据
UART_RecBuf[rec_index] = rec_data;
rec_index++;
TH0 = 0xB7;
TL0 = 0xFF;
RI = 0; //清除接收中断标志位
}
}
void tim0() interrupt 1{
TL0 = 0xFF;
TH0 = 0xB7;
if(rec_index >0)
{
//协议
if(UART_RecBuf[0] == '5' & UART_RecBuf[1] == '5') //这里是接收到文本模式的55返回温度
{
unsigned int i = 0;
while(i < 5)
{
delay(1000);
display();
i++;
if(i > 5)
{
i = 0;
break;
}
}
}
rec_index=0;
}
}
//串口发送一个字节数据
void UART_SendByte(u8 Byte)
{
SBUF=Byte;
//检测是否完成
while(TI==0);
TI=0;//TI复位
}
void UART_SendBytes(u8 *Byte, u8 len)
{
char i;
for(i=0;i<len;i++)
{
UART_SendByte(*(Byte+i));
}
}
void UART_sendString(u8 *str)
{
while(*str!='\0')
{
UART_SendByte(*str);
str++;
}
}
定时器使用了中断,所以主要内容还是在这了。
uart.h
#ifndef __UART_H__
#define __UART_H__
#include "main.h"
#include "stdio.h"
void UART_Init(u8 baud);
void tim_Init();
void uart();
void tim0();
void UART_SendByte(u8 Byte);
void UART_sendString(u8 *str);
void UART_SendBytes(u8 *Byte, u8 len);
#endif //__UART_H__
原本应该将串口和定时器分开为不同的模块写的。但这里偷懒,所以放一起了。
二、串口助手
因为定义的是发送55会循环返回温度几次,这里就展示这么多。
结尾
#一课一得#