第七届蓝桥杯全国软件和信息技术专业人才大赛个人赛(电子类省赛)

该程序实现了基于STM32单片机对DS18B20温度传感器的数据读取和显示,通过单总线通信协议进行交互。代码中包含了初始化、定时器配置、PWM电机控制、数码管显示和按键扫描等功能,能够实时显示温度并根据用户操作调整电机工作模式和时间设定。
摘要由CSDN通过智能技术生成

 

 

 

 由于比较简单就没有过多模块化,个人习惯备注写在胖边·~~~~~~

main.c

#include <reg52.h>
#include "ds18.h"
#define ur unsigned char
#define ut unsigned int

sfr AUXR=0X8E;
sfr IE2=0XAF;
sfr T2H=0XD6;
sfr T2L=0XD7;
sbit P34=P3^4;

ut time1=50;
ur k=1;
ur mode=0;

int T;         //温度
int time[]={0,60,120};
int i=0;
int timed;

void hc138(ur pos,ur dat);
void SMGdisplay();
void Timer2Init();
void Timer1Init(void);	
void Scankeys();
void C_SMG();
void C_key();
void M_funtion();
ur rd_temperature();
//==============主函数============================
void main(){

	hc138(5,0x00);
    Timer2Init();
    Timer1Init();
	timed=time1;
	while(1)
		{
			T=rd_temperature()*0.0625;
			C_key();
			SMGdisplay();
			Scankeys();
			M_funtion();
         }
}


//=======================基础===============================
	

void hc138(ur pos,ur dat)
{
	switch(pos)
	{
	case 4:P0=dat;P2=0x1f&P2|0x80;break;
	case 5:P0=dat;P2=0x1f&P2|0xa0;break;
	case 6:P0=dat;P2=0x1f&P2|0xc0;break;
	case 7:P0=dat;P2=0x1f&P2|0xe0;break;
	}
	P2&=0x1f;
}


void Timer1Init(void)		//100微秒@11.0592MHz
{
	AUXR &= 0xBF;		//定时器时钟12T模式
	TMOD &= 0x0F;		//设置定时器模式
	TL1 = 0xA4;		//设置定时初值
	TH1 = 0xFF;		//设置定时初值
	TF1 = 0;		//清除TF1标志
	TR1 = 1;		//定时器1开始计时
	ET1=1;
	EA=1;
}

ur pwm_count=0;
ur pwm_duty;
void Timer1()interrupt 3
{
	pwm_count++;
	if(pwm_count < pwm_duty)
	{
		P34 = 1;
	}
		
	if(pwm_count > pwm_duty)
	{
		P34 = 0;
	}
		
	if(pwm_count >= 10)
	{
		pwm_count = 0;
		P34 = 1;
	}
}

void Timer2Init(void)		//2000微秒@11.0592MHz
{
	AUXR &= 0xFB;		//定时器时钟12T模式
	T2L = 0xCD;		//设置定时初值
	T2H = 0xF8;		//设置定时初值
	AUXR |= 0x10;		//定时器2开始计时
	IE2|=0x04;
	EA=1;
}

void Timer2()interrupt 12
{
   C_SMG();
   i++;
   if(i>=500)   //表示当够1s时要干的事情
   {
	   i=0;
       timed--;   
   if(timed<=0)
      timed=0;
  }
}
//=======================PWM电机=======================================
void M_funtion()
{
	if(time[k] > 0)  //时间大于零电机转动
	{
		TR1 = 1;
		if(mode == 0)
		{
			pwm_duty = 2;
		}
		else if(mode == 1)
		{
			pwm_duty = 3;
		}
		else if(mode == 2)
		{
			pwm_duty = 7;
		}
		
	}
	else
	{
		TR1 = 0;
		pwm_duty = 0;
		
	}
}
//========================DS18B20================================
ur rd_temperature()
{
	ut dat;
	ur LSB,MSB;
	init_ds18b20();           //DS18B20复位
	Write_DS18B20(0xcc);      //跳过ROM指令
	Write_DS18B20(0x44);      //开始温度转换
	Delay_OneWire(1000);       //延时700ms左右等待温度转换完成
	
	init_ds18b20();           //DS18B20复位
	Write_DS18B20(0xcc);      //跳过ROM指令
	Write_DS18B20(0xbe);      //开始读取高速暂存器
	LSB=Read_DS18B20();       //读取温度数据的低8位
	MSB=Read_DS18B20();       //读取温度数据的高8位
	init_ds18b20();           //DS18B20复位
	
	dat=0x0000;
	dat=MSB;
    dat=(MSB<<8)|LSB;
	
	if((dat&0xf800)==0x0000)   //如果温度为负
	{
		dat>>=4;
		dat=dat*10;
		dat=dat+(LSB&0x0f)*0.625;
		
	}
    return dat;           //将LSB与MSB整合成一个16位的整数
	
}

//=============================数码管-LED==========================
ur SMGNodot_ca[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,
	0x80,0x90,0xbf,0xff,0xc6};          //0-9  -  关闭   C 
ur disbuf[]={11,11,11,11,11,11,11,11};  //数码管全部为关闭状态
ur posd=0;
ur interface=0;

void C_SMG()          //数码管-控制
{
	hc138(7,0xff);        //消隐
	hc138(6,0x01<<posd);  //控制位
	hc138(7,SMGNodot_ca[disbuf[posd]]);
    posd++;
     if(posd>=8)
        posd=0;
}

void SMGdisplay()
{
  if(interface==0)   //温度显示界面
  {
	disbuf[0]=disbuf[2]=10;
	disbuf[1]=mode+1;
	disbuf[3]=11;
	disbuf[4]=timed/1000;
	disbuf[5]=timed/100%10;
	disbuf[6]=timed/10%10;
	disbuf[7]=timed%10;
  }
  if(interface==1)  //时间显示界面
	  {
		disbuf[0]=disbuf[2]=10;
	    disbuf[1]=4;
	    disbuf[3]=disbuf[4]=11;
		disbuf[5]=T/10;
	    disbuf[6]=T%10;
		disbuf[7]=12;
	  }
	if(mode==0)
	hc138(4,0xfe);//L1灯亮
	else if(mode==1)
	hc138(4,0xfd);//L2灯亮
	else if(mode==2)
	hc138(4,0xfb);//L3灯亮
}

//=================================按键========================
ur ked=0;
char set=-1;
void delay(ut n)
{
	while(n--);
}

void C_key()
{  
      if(((P3&0x0f)!=0x0f)&&(ked==0))//有键按下
		  {
        delay(2);
           if(((P3&0x0f)!=0x0f)&&(ked==0))
	           {
	              ked=1;
	              switch(P3&0x0f)
		         {
	              case 0x0e:set=0;break;//s7
	              case 0x0d:set=1;break;//s6
	              case 0x0b:set=2;break;//s5
	              case 0x07:set=3;break;//s4
	    }
               }
	     }
       else if(((P3&0x0f)==0x0f)&&(ked==1))
       ked=0;
}

void Scankeys()
{
		if(set==3)      //原本状态为set=-1,只有s4按下为3,又初始化为-1。
			{
				set=-1;
				mode++;   //mode初状态为0,界面为-1-  
				if(mode>=3)
					mode=0;
			}
		if(set==2)
			{
				set=-1;
				k++;       //k初状态为时间0,
				if(k>=3)
					k=0;
				timed=time[k];
			}
		if(set==1)
			{
				set=-1;
				timed=0;
				k=0;
				hc138(4,0xff);
			}
		if(set==0)
			{
				set=-1;
				if(interface==0)
					interface=1;
				else
					interface=0;
			}
 }

ds18.c

/*
  程序说明: 单总线驱动程序
  软件环境: Keil uVision 4.10 
  硬件环境: CT107单片机综合实训平台(外部晶振12MHz) STC89C52RC单片机
  日    期: 2011-8-9
*/
#include "reg52.h"
#include "ds18.h"
sbit DQ = P1^4;  //单总线接口

//单总线延时函数
void Delay_OneWire(unsigned int t)  //STC89C52RC
{	 int i;
	while(t--)
	for(i=0;i<10;i++);
}

//通过单总线向DS18B20写一个字节
void Write_DS18B20(unsigned char dat)
{
	unsigned char i;
	for(i=0;i<8;i++)
	{
		DQ = 0;
		DQ = dat&0x01;
		Delay_OneWire(5);
		DQ = 1;
		dat >>= 1;
	}
	Delay_OneWire(5);
}

//从DS18B20读取一个字节
unsigned char Read_DS18B20(void)
{
	unsigned char i;
	unsigned char dat;
  
	for(i=0;i<8;i++)
	{
		DQ = 0;
		dat >>= 1;
		DQ = 1;
		if(DQ)
		{
			dat |= 0x80;
		}	    
		Delay_OneWire(5);
	}
	return dat;
}

//DS18B20设备初始化
bit init_ds18b20(void)
{
  	bit initflag = 0;
  	
  	DQ = 1;
  	Delay_OneWire(12);
  	DQ = 0;
  	Delay_OneWire(80);
  	DQ = 1;
  	Delay_OneWire(10); 
    initflag = DQ;     
  	Delay_OneWire(5);
  
  	return initflag;
}

ds18.h

#ifndef __DS18_H
#define __DS18_H

void Delay_OneWire(unsigned int t);
void Write_DS18B20(unsigned char dat);
unsigned char Read_DS18B20(void);
bit init_ds18b20(void);

#endif

完成如下

 

 

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值