用温度芯片ds18b20编写简单的检测温度程序,可总是不能正常运行。数码管总是静止显示271,请大虾指点。

用温度芯片ds18b20编写简单的检测温度程序,可总是不能正常运行。数码管总是静止显示271,请大虾指点。,第1张

ds18b20加lcd1602加串口输出最近刚刚做的,调试成功了,lcddata口是P0

#include<intrins.h>

#include<stdio.h>

#include"STC12C5A.H"

#define XTAL 11059200 // CUP 晶振频率

#define baudrate 9600 /判察/ 通信波特率

#define uchar unsigned char

#define uint unsigned int

sbit DQ=P2^3 //DS18B20数据

sbit E=P2^7 //1602使能

sbit RW=P2^6 //1602读写

sbit RS=P2^5 //1602数据/命令选择

uchar data temp_data[2] = {0x00,0x00} //高低8位数据缓存

uchar code num[]={'掘让茄0'滑郑,'1','2','3','4','5','6','7','8','9'} //lcd1602数字转ASCII表用于显示数字

uint DS[4] //温度显示4位有效数字

uint dh

uint dl

uint DStemp //16进制转10进制数据缓存

void Delay(unsigned int num) { // DS18B20及主程序延时函数 1=3us

unsigned int i

for(i=numi>0i--)

}

void InitUART(void) //串口初始化

{

TMOD = 0x20

SCON = 0x50

TH1 = 0xFD

TL1 = TH1

PCON = 0x00

EA = 1

ES = 1

TR1 = 1

}

void SendOneByte(unsigned char c) //串口发送1字节

{

SBUF = c

while(!TI)

TI = 0

}

void delay() //lcd1602延迟

{

int i,j

for(i=0i<=50i++)

for(j=0j<=2j++)

}

void wcmd(uchar del) //命令写入程序

{

P0 = del

RS = 0

RW = 0

E = 0

delay()

E = 1

delay()

}

void write(uchar del) //写入一位数据

{

P0 = del

RS = 1

RW = 0

E = 0

delay()

E = 1

delay()

}

void L1602_init(void) //lcd1602初始化

{

wcmd(0x01)

wcmd(0x38)

wcmd(0x0c)

wcmd(0x06)

wcmd(0xd0)

}

void L1602_char(uchar hang,uchar lie,char sign) //写入1个字符

{

uchar a

if(hang == 1) a = 0x80

if(hang == 2) a = 0xc0

a = a + lie - 1

wcmd(a)

write(sign)

}

void L1602_string(uchar hang,uchar lie,uchar *p) //写入字符串

{

uchar a,b=0

if(hang == 1) a = 0x80

if(hang == 2) a = 0xc0

a = a + lie - 1

while(1)

{

wcmd(a++)

b++

if((*p == '\0')||(b==16)) break

write(*p)

p++

}

}

uchar Init_DS18B20(void) { // 初始化ds1820

DQ=1// DQ复位

Delay(22)// 稍做延时

DQ=0// 单片机将DQ拉低

Delay(160)// 精确延时 大于 480us

DQ=1// 拉高总线

Delay(22)

while(DQ)

L1602_string(1,15,"ok") //初始化成功在LCD1602上显示OK

Delay(166)

DQ=1

return(1)// 返回信号,0=presence,1= no presence

}

uchar ReadOneChar(void) { // 读一个字节

uchar i

uchar 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

Delay(22)//66 us

}

DQ=1

return(value)

}

void WriteOneChar(unsigned char dat) { // 写一个字节

unsigned char i=0

for(i=8i>0i--) {

DQ=0

DQ=dat&0x01

Delay(18)

DQ=1

dat>>=1

}

}

void Read_Temperature(void) { // 读取温度

Init_DS18B20()

WriteOneChar(0xCC)// 跳过读序号列号的 *** 作

WriteOneChar(0x44)// 启动温度转换

Init_DS18B20()

WriteOneChar(0xCC)// 跳过读序号列号的 *** 作

WriteOneChar(0xBE)// 读取温度寄存器

temp_data[0]=ReadOneChar()// 温度低8位

temp_data[1]=ReadOneChar()// 温度高8位

}

void Main()

{

L1602_init() //初始化LCD1602

Init_DS18B20() //初始化DS18B20

InitUART() //初始化串口通信

L1602_string(1,1,"DS18B20") //lcd1602测试

L1602_string(2,15,"`c")

while(1)

{

Read_Temperature() //读取高低8位数据

Delay(800000)

dl=temp_data[0]

dh=temp_data[1]

SendOneByte(dh) //串口发送高低8位数据

SendOneByte(dl)

DStemp=(dh*256+dl)/10*625/10 //16进制转换10进制 先除10否则会溢出

DS[0]=DStemp%10 //提取温度各个位数字

DS[1]=DStemp/10%10

DS[2]=DStemp/100%10

DS[3]=DStemp/1000

L1602_char(1,3,'.') //显示各个位数字

L1602_char(1,5,num[DS[0]])

L1602_char(1,4,num[DS[1]])

L1602_char(1,2,num[DS[2]])

L1602_char(1,1,num[DS[3]])

}

}

其实转ascii码只要加38就好了,当时没想到

要求用51单片机做一个风扇温控检测系统,其功能大概就锋凳是:当温度达到设定值时候,风扇电机开始转动,以达到散热的目的,题目本身没有什么意思,有意思的慎基洞是想要用尽可能少的元器件来完成这项工作,要求使用两个数码管,其他没说,所以我就加了3个按键,大体思路是这样的,开机以后,正常显示温度,按下SET键位时候,进温度设置,两个按键+和-设置数值。再按一次SET保存并退出。

最有意思的在于用了两个标志位,这两个标志位简直没把我绕晕。现在完成了,发出来大家一起看看,不在于代码多好,在于一种解决问题的方法。我把文件发上来了,大家一起学习交流。

仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)

TIM截图20181216211410.jpg TIM截图20181216211656.jpg TIM截图20181216211828.jpg

单片机源程序如下:

代码:

#include "reg52.h"

#include <intrins.h>

#define _Nop() _nop_()//宽枯定义空指令

/*-----------------------------------------------

IO口定义

-----------------------------------------------*/

sbit LED_R = P1^6//红灯

sbit BEEP = P2^1 //蜂鸣器

sbit MOTOR = P2^2//电机

sbit KEY_ADD = P2^5 //加

sbit KEY_DEC = P2^4 //减

sbit KEY_SET = P2^3 //确认

sbit ONE = P2^6 //个位位选

sbit TEN = P2^7 //十位位选

sbit DQ=P1^7 //DS18B20数据

sbit SDA=P1^1 //模拟I2C数据传送位

sbit SCL=P1^0 //模拟I2C时钟控制位

/*-----------------------------------------------

定义变量

-----------------------------------------------*/

unsigned char Set_Flag=0 //

unsigned char Set_Value=40//设定值

unsigned char Tempature = 0//温度

unsigned char num=40 //数字

unsigned char table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f}//数码管段选

unsigned char Data=0

char Temp[12]

bit ack //应答标志位

/*-----------------------------------------------

延时函数

-----------------------------------------------*/

void Delay_us(unsigned char t)

{

while(t--)

}

void Delay_ms(unsigned char t)

{

int i,j

for(i=ti>0i--)

for(j=125j>0j--)

}

void delay_18B20(unsigned int i)//微秒级延时

{

while(i--)

}

/*-----------------------------------------------

DS18B20初始化

-----------------------------------------------*/

void ds1820rst(void)/*ds1820复位*/

{

unsigned char x=0

DQ = 1//DQ复位

delay_18B20(5)//延时

DQ = 0//DQ拉低

delay_18B20(100)//精确延时大于480us

DQ = 1//拉高

delay_18B20(40)

}

unsigned char ds1820rd(void)//从18b20读取数据

{

unsigned char i=0

unsigned char dat = 0

for (i=8i>0i--)

{

DQ = 0//给脉冲信号

dat>>=1

DQ = 1//给脉冲信号

if(DQ) dat|=0x80

delay_18B20(10)

}

return(dat)

}

void ds1820wr(unsigned char wdata)/*写数据*/

{

unsigned char i=0

for (i=8i>0i--)

{

DQ = 0

DQ = wdata&0x01

delay_18B20(25)

DQ = 1

wdata>>=1

}

}

//读取DS18B20温度值

//入口参数:无

//返回值:温度值

unsigned int read_18b20(void)/*读取温度值并转换*/

{

unsigned char a,b

unsigned char tem

ds1820rst()

ds1820wr(0xcc)//*跳过读序列号*/

ds1820wr(0x44)//*启动温度转换*/

ds1820rst()

ds1820wr(0xcc)//*跳过读序列号*/

ds1820wr(0xbe)//*读取温度*/

a=ds1820rd()

b=ds1820rd()

tem=((b<<4)|(a>>4)) //将高低字节组合成字

//Delay_ms(50)

return tem //完成温度转换

}

/*-----------------------------------------------

数码管

-----------------------------------------------*/

//共阴数码管

//动态扫描

void display(unsigned char Data)

{

//P3 = 0

ONE = 1//打开个位位选

TEN = 0

P3 = table[Data/10]//送个位

Delay_ms(5)//延时

P3 = 0 //防止重影

//Delay_us(5)

TEN = 1

ONE = 0

P3 = table[Data%10]

Delay_ms(5)

P3 = 0

//Delay_us(5)

}

/*-----------------------------------------------

按键处理函数

-----------------------------------------------*/

char S_F=0

unsigned char Key_Value(void)

{

if(!KEY_ADD || !KEY_DEC||!KEY_SET) // 检测是否有任意按键按下

{

Delay_ms(20)

Set_Flag = 1 //切换到显示设置值

if(!KEY_ADD) //如果检测到低电平,说明按键按下

{

Delay_ms(10)//延时去抖,一般10-20ms

if(!KEY_ADD) //再次确认按键是否按下,没有按下则退出

{

while(!KEY_ADD)//如果确认按下按键等待按键释放,没有释放则一直等待

{

if(num<99)//加 *** 作

num++

}

}

}

if(!KEY_DEC) //如果检测到低电平,说明按键按下

{

Delay_ms(10)//延时去抖,一般10-20ms

if(!KEY_DEC) //再次确认按键是否按下,没有按下则退出

{

while(!KEY_DEC)//如果确认按下按键等待按键释放,没有释放则一直等待

{

if(num>0) //减 *** 作

num--

}

}

}

if(!KEY_SET) //如果检测到低电平,说明按键按下

{

Delay_ms(10)//延时去抖,一般10-20ms

if(!KEY_SET) //再次确认按键是否按下,没有按下则退出

{

//Set_Flag = 0

while(!KEY_SET)//如果确认按下按键等待按键释放,没有释放则一直等待

{

//将当前值作为设定值

//Set_Flag = 0 //切换到显示温度值

}

if(S_F)

Set_Flag=0

}

}

}

Set_Value = num

return (num)

}

/*-----------------------------------------------

24C02初始化

-----------------------------------------------*/

/*------------------------------------------------

启动总线

------------------------------------------------*/

void Start_I2c()

{

SDA=1 //发送起始条件的数据信号

_Nop()

SCL=1

_Nop() //起始条件建立时间大于4.7us,延时

_Nop()

_Nop()

_Nop()

_Nop()

SDA=0//发送起始信号

_Nop() //起始条件锁定时间大于4μ

_Nop()

_Nop()

_Nop()

_Nop()

SCL=0 //钳住I2C总线,准备发送或接收数据

_Nop()

_Nop()

}

/*------------------------------------------------

结束总线

------------------------------------------------*/

void Stop_I2c()

{

SDA=0 //发送结束条件的数据信号

_Nop() //发送结束条件的时钟信号

SCL=1 //结束条件建立时间大于4μ

_Nop()

_Nop()

_Nop()

_Nop()

_Nop()

SDA=1 //发送I2C总线结束信号

_Nop()

_Nop()

_Nop()

_Nop()

}

/*main.c  伍御 自己参考一下 改改吧

#include<reg51.h>

#include"DS18B20_3.H"

#define uint unsigned int

#define uchar unsigned char

uchar a,miao,shi,fen,ri,yue,nian,week,flag,key1n,temp

//flag用于读取头文件中的温度值,和显示温度值

#define yh 0x80 //LCD第一行的初始位置,因为LCD1602字符地址首位D7恒定为1(100000000=80)

#define er 0x80+0x40 //LCD第二行初始位置(因为第二行第一个字符位置地址是0x40)

//液晶屏的与C51之间的引脚连接定义(显示数据线接C51的P0口)

sbit rs=P2^0

sbit en=P2^2

sbit rw=P2^1 //如果硬件上rw接地,就不用写这句和后面的rw=0了

sbit led=P2^6 //LCD背光开关

//DS1302时钟芯片与腔友岩C51之间的引脚连接定义

sbit IO=P1^1

sbit SCLK=P1^0

sbit RST=P1^2

sbit ACC0=ACC^0

sbit ACC7=ACC^7

/************************************************************

ACC累加器=A

ACC.0=E0H

ACC.0就是ACC的第0位。Acc可以位寻址。

累加器ACC是一个8位的存储单元,是用来放数据的。但是,这个存储单元有其特殊的地位,

是单片机中一个非常关键的单元,很多运算都要通告肆过ACC来进行。以后在学习指令时,

常用A来表示累加器。但有一些地方例外,比如在PUSH指令中,就必须用ACC这样的名字。

一般的说法,A代表了累加器中的内容、而ACC代表的是累加器的地址。

***************************************************************/

//校时按键与C51的引脚连接定义

sbit key1=P1^5  //设置键

sbit key2=P1^6  //加键

sbit key3=P1^7  //减键

sbit buzzer=P1^3//蜂鸣器,通过三极管9012驱动,端口低电平响

/**************************************************************/

uchar code tab1[]={"20  -  -   "} //年显示的固定字符

uchar code tab2[]={"  :  :  "}//时间显示的固定字符

//延时函数,后面经常调用

void delay(uint xms)//延时函数,有参函数

{

uint x,y

for(x=xmsx>0x--)

for(y=110y>0y--)

}

/********液晶写入指令函数与写入数据函数,以后可调用**************/

/*在这个程序中,液晶写入有关函数会在DS1302的函数中调用,所以液晶程序要放在前面*/

write_1602com(uchar com)//****液晶写入指令函数****

{

rs=0//数据/指令选择置为指令

rw=0 //读写选择置为写

P0=com//送入数据

delay(1)

en=1//拉高使能端,为制造有效的下降沿做准备

delay(1)

en=0//en由高变低,产生下降沿,液晶执行命令

}

write_1602dat(uchar dat)//***液晶写入数据函数****

{

rs=1//数据/指令选择置为数据

rw=0 //读写选择置为写

P0=dat//送入数据

delay(1)

en=1 //en置高电平,为制造下降沿做准备

delay(1)

en=0 //en由高变低,产生下降沿,液晶执行命令

}

lcd_init()//***液晶初始化函数****

{

write_1602com(0x38)//设置液晶工作模式,意思:16*2行显示,5*7点阵,8位数据

write_1602com(0x0c)//开显示不显示光标

write_1602com(0x06)//整屏不移动,光标自动右移

write_1602com(0x01)//清显示

write_1602com(yh+1)//日历显示固定符号从第一行第1个位置之后开始显示

for(a=0a<14a++)

{

write_1602dat(tab1[a])//向液晶屏写日历显示的固定符号部分

//delay(3)

}

write_1602com(er+2)//时间显示固定符号写入位置,从第2个位置后开始显示

for(a=0a<8a++)

{

write_1602dat(tab2[a])//写显示时间固定符号,两个冒号

//delay(3)

}

}

/*********************over***********************/

/***************DS1302有关子函数********************/

void write_byte(uchar dat)//写一个字节

{

ACC=dat

RST=1

for(a=8a>0a--)

{

IO=ACC0

SCLK=0

SCLK=1

ACC=ACC>>1

}

}

uchar read_byte()//读一个字节

{

RST=1

for(a=8a>0a--)

{

ACC7=IO

SCLK=1

SCLK=0

ACC=ACC>>1

}

return (ACC)

}

//----------------------------------------

void write_1302(uchar add,uchar dat)//向1302芯片写函数,指定写入地址,数据

{

RST=0

SCLK=0

RST=1

write_byte(add)

write_byte(dat)

SCLK=1

RST=0

}

uchar read_1302(uchar add)//从1302读数据函数,指定读取数据来源地址

{

uchar temp

RST=0

SCLK=0

RST=1

write_byte(add)

temp=read_byte()

SCLK=1

RST=0

return(temp)

}

uchar BCD_Decimal(uchar bcd)//BCD码转十进制函数,输入BCD,返回十进制

{

uchar Decimal

Decimal=bcd>>4

return(Decimal=Decimal*10+(bcd&=0x0F))

}

//--------------------------------------

void ds1302_init() //1302芯片初始化子函数(2010-01-07,12:00:00,week4)

{

RST=0

SCLK=0

write_1302(0x8e,0x00) //允许写,禁止写保护

//write_1302(0x80,0x00) //向DS1302内写秒寄存器80H写入初始秒数据00

//write_1302(0x82,0x00)//向DS1302内写分寄存器82H写入初始分数据00

//write_1302(0x84,0x12)//向DS1302内写小时寄存器84H写入初始小时数据12

//write_1302(0x8a,0x04)//向DS1302内写周寄存器8aH写入初始周数据4

//write_1302(0x86,0x07)//向DS1302内写日期寄存器86H写入初始日期数据07

//write_1302(0x88,0x01)//向DS1302内写月份寄存器88H写入初始月份数据01

//write_1302(0x8c,0x10)//向DS1302内写年份寄存器8cH写入初始年份数据10

write_1302(0x8e,0x80) //打开写保护

}

//------------------------------------

//温度显示子函数

void write_temp(uchar add,uchar dat)//向LCD写温度数据,并指定显示位置

{

uchar gw,sw

gw=dat%10//取得个位数字

sw=dat/10//取得十位数字

write_1602com(er+add)//er是头文件规定的值0x80+0x40

write_1602dat(0x30+sw)//数字+30得到该数字的LCD1602显示码

write_1602dat(0x30+gw)//数字+30得到该数字的LCD1602显示码

write_1602dat(0xdf)//显示温度的小圆圈符号,0xdf是液晶屏字符库的该符号地址码

write_1602dat(0x43) //显示"C"符号,0x43是液晶屏字符库里大写C的地址码

}

//------------------------------------

//时分秒显示子函数

void write_sfm(uchar add,uchar dat)//向LCD写时分秒,有显示位置加、现示数据,两个参数

{

uchar gw,sw

gw=dat%10//取得个位数字

sw=dat/10//取得十位数字

write_1602com(er+add)//er是头文件规定的值0x80+0x40

write_1602dat(0x30+sw)//数字+30得到该数字的LCD1602显示码

write_1602dat(0x30+gw)//数字+30得到该数字的LCD1602显示码

}

//-------------------------------------

//年月日显示子函数

void write_nyr(uchar add,uchar dat)//向LCD写年月日,有显示位置加数、显示数据,两个参数

{

uchar gw,sw

gw=dat%10//取得个位数字

sw=dat/10//取得十位数字

write_1602com(yh+add)//设定显示位置为第一个位置+add

write_1602dat(0x30+sw)//数字+30得到该数字的LCD1602显示码

write_1602dat(0x30+gw)//数字+30得到该数字的LCD1602显示码

}

//-------------------------------------------

void write_week(uchar week)//写星期函数

{

write_1602com(yh+0x0c)//星期字符的显示位置

switch(week)

{

case 1:write_1602dat('M')//星期数为1时,显示

write_1602dat('O')

write_1602dat('N')

break

case 2:write_1602dat('T')//星期数据为2时显示

write_1602dat('U')

write_1602dat('E')

break

case 3:write_1602dat('W')//星期数据为3时显示

write_1602dat('E')

write_1602dat('D')

break

case 4:write_1602dat('T')//星期数据为4是显示

write_1602dat('H')

write_1602dat('U')

break

case 5:write_1602dat('F')//星期数据为5时显示

write_1602dat('R')

write_1602dat('I')

break

case 6:write_1602dat('S')//星期数据为6时显示

write_1602dat('T')

write_1602dat('A')

break

case 7:write_1602dat('S')//星期数据为7时显示

write_1602dat('U')

write_1602dat('N')

break

}

}

//****************键盘扫描有关函数**********************

void keyscan()

{

if(key1==0)//---------------key1为功能键(设置键)--------------------

{

delay(9)//延时,用于消抖动

if(key1==0)//延时后再次确认按键按下

{

buzzer=0//蜂鸣器短响一次

delay(20)

buzzer=1

while(!key1)

key1n++

if(key1n==9)

key1n=1//设置按键共有秒、分、时、星期、日、月、年、返回,8个功能循环

switch(key1n)

{

case 1: TR0=0//关闭定时器

//TR1=0

write_1602com(er+0x09)//设置按键按动一次,秒位置显示光标

write_1602com(0x0f)//设置光标为闪烁

temp=(miao)/10*16+(miao)%10//秒数据写入DS1302

write_1302(0x8e,0x00)

write_1302(0x80,0x80|temp)//miao

write_1302(0x8e,0x80)

break

case 2:  write_1602com(er+6)//按2次fen位置显示光标

//write_1602com(0x0f)

break

case 3: write_1602com(er+3)//按动3次,shi

//write_1602com(0x0f)

break

case 4: write_1602com(yh+0x0e)//按动4次,week

//write_1602com(0x0f)

break

case 5: write_1602com(yh+0x0a)//按动5次,ri

//write_1602com(0x0f)

break

case 6: write_1602com(yh+0x07)//按动6次,yue

//write_1602com(0x0f)

break

case 7: write_1602com(yh+0x04)//按动7次,nian

//write_1602com(0x0f)

break

case 8:

write_1602com(0x0c)//按动到第8次,设置光标不闪烁

TR0=1//打开定时器

temp=(miao)/10*16+(miao)%10

write_1302(0x8e,0x00)

write_1302(0x80,0x00|temp)//miao数据写入DS1302

write_1302(0x8e,0x80)

break

}

}

}

//------------------------------加键key2----------------------------

if(key1n!=0)//当key1按下以下。再按以下键才有效(按键次数不等于零)

{

if(key2==0)  //上调键

{

delay(10)

if(key2==0)

{

buzzer=0//蜂鸣器短响一次

delay(20)

buzzer=1

while(!key2)

switch(key1n)

{

case 1:miao++//设置键按动1次,调秒

if(miao==60)

miao=0//秒超过59,再加1,就归零

write_sfm(0x08,miao)//令LCD在正确位置显示"加"设定好的秒数

temp=(miao)/10*16+(miao)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00) //允许写,禁止写保护

write_1302(0x80,temp) //向DS1302内写秒寄存器80H写入调整后的秒数据BCD码

write_1302(0x8e,0x80) //打开写保护

write_1602com(er+0x09)//因为设置液晶的模式是写入数据后,光标自动右移,所以要指定返回

//write_1602com(0x0b)

break

case 2:fen++

if(fen==60)

fen=0

write_sfm(0x05,fen)//令LCD在正确位置显示"加"设定好的分数据

temp=(fen)/10*16+(fen)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00)//允许写,禁止写保护

write_1302(0x82,temp)//向DS1302内写分寄存器82H写入调整后的分数据BCD码

write_1302(0x8e,0x80)//打开写保护

write_1602com(er+6)//因为设置液晶的模式是写入数据后,指针自动加一,在这里是写回原来的位置

break

case 3:shi++

if(shi==24)

shi=0

write_sfm(2,shi)//令LCD在正确的位置显示"加"设定好的小时数据

temp=(shi)/10*16+(shi)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00)//允许写,禁止写保护

write_1302(0x84,temp)//向DS1302内写小时寄存器84H写入调整后的小时数据BCD码

write_1302(0x8e,0x80)//打开写保护

write_1602com(er+3)//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位

break

case 4:week++

if(week==8)

week=1

write_1602com(yh+0x0C)//指定'加'后的周数据显示位置

write_week(week)//指定周数据显示内容

temp=(week)/10*16+(week)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00)//允许写,禁止写保护

write_1302(0x8a,temp)//向DS1302内写周寄存器8aH写入调整后的周数据BCD码

write_1302(0x8e,0x80)//打开写保护

write_1602com(yh+0x0e)//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位

break

case 5:ri++

if(ri==32)

ri=1

write_nyr(9,ri)//令LCD在正确的位置显示"加"设定好的日期数据

temp=(ri)/10*16+(ri)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00)//允许写,禁止写保护

write_1302(0x86,temp)//向DS1302内写日期寄存器86H写入调整后的日期数据BCD码

write_1302(0x8e,0x80)//打开写保护

write_1602com(yh+10)//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位

break

case 6:yue++

if(yue==13)

yue=1

write_nyr(6,yue)//令LCD在正确的位置显示"加"设定好的月份数据

temp=(yue)/10*16+(yue)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00)//允许写,禁止写保护

write_1302(0x88,temp)//向DS1302内写月份寄存器88H写入调整后的月份数据BCD码

write_1302(0x8e,0x80)//打开写保护

write_1602com(yh+7)//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位

break

case 7:nian++

if(nian==100)

nian=0

write_nyr(3,nian)//令LCD在正确的位置显示"加"设定好的年份数据

temp=(nian)/10*16+(nian)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00)//允许写,禁止写保护

write_1302(0x8c,temp)//向DS1302内写年份寄存器8cH写入调整后的年份数据BCD码

write_1302(0x8e,0x80)//打开写保护

write_1602com(yh+4)//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位

break

}

}

}

//------------------减键key3,各句功能参照'加键'注释---------------

if(key3==0)

{

delay(10)//调延时,消抖动

if(key3==0)

{

buzzer=0//蜂鸣器短响一次

delay(20)

buzzer=1

while(!key3)

switch(key1n)

{

case 1:miao--

if(miao==-1)

miao=59//秒数据减到-1时自动变成59

write_sfm(0x08,miao)//在LCD的正确位置显示改变后新的秒数

temp=(miao)/10*16+(miao)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00) //允许写,禁止写保护

write_1302(0x80,temp) //向DS1302内写秒寄存器80H写入调整后的秒数据BCD码

write_1302(0x8e,0x80) //打开写保护

write_1602com(er+0x09)//因为设置液晶的模式是写入数据后,指针自动加一,在这里是写回原来的位置

//write_1602com(0x0b)

break

case 2:fen--

if(fen==-1)

fen=59

write_sfm(5,fen)

temp=(fen)/10*16+(fen)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00)//允许写,禁止写保护

write_1302(0x82,temp)//向DS1302内写分寄存器82H写入调整后的分数据BCD码

write_1302(0x8e,0x80)//打开写保护

write_1602com(er+6)//因为设置液晶的模式是写入数据后,指针自动加一,在这里是写回原来的位置

break

case 3:shi--

if(shi==-1)

shi=23

write_sfm(2,shi)

temp=(shi)/10*16+(shi)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00)//允许写,禁止写保护

write_1302(0x84,temp)//向DS1302内写小时寄存器84H写入调整后的小时数据BCD码

write_1302(0x8e,0x80)//打开写保护

write_1602com(er+3)//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位

break

case 4:week--

if(week==0)

week=7

write_1602com(yh+0x0C)//指定'加'后的周数据显示位置

write_week(week)//指定周数据显示内容

temp=(week)/10*16+(week)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00)//允许写,禁止写保护

write_1302(0x8a,temp)//向DS1302内写周寄存器8aH写入调整后的周数据BCD码

write_1302(0x8e,0x80)//打开写保护

write_1602com(yh+0x0e)//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位

break

case 5:ri--

if(ri==0)

ri=31

write_nyr(9,ri)

temp=(ri)/10*16+(ri)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00)//允许写,禁止写保护

write_1302(0x86,temp)//向DS1302内写日期寄存器86H写入调整后的日期数据BCD码

write_1302(0x8e,0x80)//打开写保护

write_1602com(yh+10)//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位

break

case 6:yue--

if(yue==0)

yue=12

write_nyr(6,yue)

temp=(yue)/10*16+(yue)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00)//允许写,禁止写保护

write_1302(0x88,temp)//向DS1302内写月份寄存器88H写入调整后的月份数据BCD码

write_1302(0x8e,0x80)//打开写保护

write_1602com(yh+7)//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位

break

case 7:nian--

if(nian==-1)

nian=99

write_nyr(3,nian)

temp=(nian)/10*16+(nian)%10//十进制转换成DS1302要求的DCB码

write_1302(0x8e,0x00)//允许写,禁止写保护

write_1302(0x8c,temp)//向DS1302内写年份寄存器8cH写入调整后的年份数据BCD码

write_1302(0x8e,0x80)//打开写保护

write_1602com(yh+4)//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位

break

}

}

}

}

}

//-------------------------------

void init() //定时器、计数器设置函数

{

TMOD=0x11 //指定定时/计数器的工作方式为3

TH0=0  //定时器T0的高四位=0

TL0=0  //定时器T0的低四位=0

EA=1  //系统允许有开放的中断

ET0=1 //允许T0中断

TR0=1 //开启中断,启动定时器

}

//*******************主函数**************************

//***************************************************

void main()

{

lcd_init()      //调用液晶屏初始化子函数

ds1302_init()   //调用DS1302时钟的初始化子函数

init()          //调用定时计数器的设置子函数

led=0           //打开LCD的背光电源

buzzer=0//蜂鸣器长响一次

delay(80)

buzzer=1

while(1)  //无限循环下面的语句:

{

keyscan()      //调用键盘扫描子函数

}

}

void timer0() interrupt 1  //取得并显示日历和时间

{

//Init_DS18B20()//温度传感器DS18b2初始化子函数,在头文件中

flag=ReadTemperature()//将18b2头文件运行返回的函数结果送到变量FLAG中,用于显示

//读取秒时分周日月年七个数据(DS1302的读寄存器与写寄存器不一样):

miao = BCD_Decimal(read_1302(0x81))

fen = BCD_Decimal(read_1302(0x83))

shi  = BCD_Decimal(read_1302(0x85))

ri  = BCD_Decimal(read_1302(0x87))

yue = BCD_Decimal(read_1302(0x89))

nian=BCD_Decimal(read_1302(0x8d))

week=BCD_Decimal(read_1302(0x8b))

//显示温度、秒、时、分数据:

write_temp(12,flag)//显示温度,从第二行第12个字符后开始显示

write_sfm(8,miao)//秒,从第二行第8个字后开始显示(调用时分秒显示子函数)

write_sfm(5,fen)//分,从第二行第5个字符后开始显示

write_sfm(2,shi)//小时,从第二行第2个字符后开始显示

//显示日、月、年数据:

write_nyr(9,ri)//日期,从第二行第9个字符后开始显示

write_nyr(6,yue)//月份,从第二行第6个字符后开始显示

write_nyr(3,nian)//年,从第二行第3个字符后开始显示

write_week(week)

}


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/12538609.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-26
下一篇 2023-05-26

发表评论

登录后才能评论

评论列表(0条)

保存