基于51单片机的DS18B20温度报警器

摘 要:单片机技术已经普及到我们生活,工作,科研,各个领域,已经成为一种比较成熟的技术,本文将介绍一种基于单片机控制的数字温度器,本温度计属于多功能温度计,可以设置上下报警温度,当温度不在设置范围内时,可以报警。

随着现代工农业技术的发展及人们对生活环境要求的提高,人们也迫切需要检测与控制温度。本文通过采用蜂鸣器作为电声元件的温度报警器的设计,阐明了该装置进行设计与制作的具体过程及方法。这种温度报警器结构简单,可操作性强,应用广泛。工作时,温度测量范围为5—38ºC。当前环境温度若超过设定的高温临界温度,由单片机发出报警信号,从而防止带来的不必要的损失。

造成高温火灾有:电气线路短路、过载、接触电阻过大等引发高温或火灾;静电产生高温或或火灾;雷电等强电侵入导致高温或火灾;最主要是机房内电脑、空调等用电设备长时间工作,导致设备老化,空调发生故障,而不能降温;因此机房内所属的电子产品发热快,在短时间内机房温度升高超出设备正常温度,导致系统瘫痪或产生火灾,这时温度报警系统就会发挥应有的功能。

关键词:STC89C51单片机,数字控制,温度计,DS18B20,24c02

前言

随着人们生活水平的不断提高,单片机控制无疑是人们追求的目标之一,它所给人带来的方便也是不可否定的,其中数字温度计就是一个典型的例子,但人们对它的要求越来越高,要为现代人工作、科研、生活、提供更好的更方便的设施就需要从数单片机技术入手,一切向着数字化控制,智能化控制方向发展。本设计所介绍的数字温度计与传统的温度计相比,具有读数方便,测温范围广,测温准确,掉电保持上下限温度值,其输出温度采用数字显示,主要用于对测温比较准确的场所,或科研实验室使用,该设计控制器使用单片机 STC89C51,测温传感器使用 DS18B20,掉电保持使用24c02,用四位一体共阳极 LED 数码管以串口传送数据,实现温度显示,能准确达到要求

1 设计要求与方案论证

首先明确设计要求,再讨论方案,一一攻破设计的难点。

1.1 设计要求

基本范围0℃-99℃ ;

精度误差小于 0.1℃ ;

数码管直读显示;

扩展功能:可以任意设定温度的上下限报警功能,并可以掉电保持上下限温度值。

1.2 系统基本方案选择和论证

1.2.1 单片机芯片的选择方案和论证

由于单片机具有以下的很多优点,被我们选定为制作该作品的首选芯片

单片机特点:

(1)高集成度,体积小,高可靠性

单片机将各功能部件集成在一块晶体芯片上,集成度很高,体积自然也是最小的。芯片本身是按工业测控环境要求设计的,内部布线很短,其抗工业噪音性能优于一般通用的CPU。单片机程序指令,常数及表格等固化在ROM中不易破坏,许多信号通道均在一个芯片内,故可靠性高。

(2)控制功能强

为了满足对对象的控制要求,单片机的指令系统均有极丰富的条件:分支转移能力,I/O口的逻辑操作及位处理能力,非常适用于专门的控制功能。

(3)低电压,低功耗,便于生产便携式产品

为了满足广泛使用于便携式系统,许多单片机内的最低工作电压仅为1.8V~3.6V,而工作电流仅为数百微安。

(4)易扩展

片内具有计算机正常运行所必需的部件。芯片外部有许多供扩展用的三总线及并行、串行输入/输出管脚,很容易构成各种规模的计算机应用系统。

(5)优异的性价比

单片机的性能极高。为了提高速度和运行效率,单片机已开始使用RISC流水线和DSP等技术。单片机的寻址能力也已突破64KB的限制,有的已可达到1MB和16MB,片内的ROM容量可达62MB,RAM容量则可达2MB。由于单片机的广泛使用,因而销量极大,各大公司的商业竞争更使其价格十分低廉,其性能价格比极高。

方案一:

采用STC89C51芯片作为硬件核心。STC89C51内部具有8KB ROM 存储空间,512字节数据存储空间,带有2K字节的EEPROM存储空间,与MCS-51系列单片机完全兼容,STC89C51可以通过串口下载。

方案二:

采用AT89S51。AT89S51片内具有8K字节程序存储空间,256字节的数据存储空间没有EEPROM存储空间,也与MCS-51系列单片机完全兼容,具有在线编程可擦除技术。

两种单片机都完全能够满足设计需要,STC89C51相对ATS89C52价格便宜,且抗干扰能力强。考虑到成本因素,因此选用STC89C51。

1.2.2 温度传感器设计方案论证

利用物质各种物理性质随温度变化的规律把温度转换为电量的传感器。这些呈现规律性变化的物理性质主要有体。温度传感器是温度测量仪表的核心部分,品种繁多。按测量方式可分为接触式和非接触式两大类,按照传感器材料及电子元件特性分为热电阻和热电偶两类。

现代信息技术的三大基础是信息采集(即传感器技术)、信息传输(通信技术)和信息处理(计算机技术)。温度传感器的发展大致经历了以下三个阶段;(1)传统的分立式温度传感器(含敏感元件);(2)模拟集成温度传感器/控制器;(3)智能温度传感器。国际上新型温度传感器正从模拟式向数字式、由集成化向智能化、网络化的方向发展。在20世纪90年代中期最早推出的智能温度传感器,采用的是8位A/D转换器,其测温精度较低,分辨力只能达到1°C。国外已相继推出多种高精度、高分辨力的智能温度传感器,所用的是9~12位A/D转换器,分辨力一般可达0.5~0.0625°C。由美国DALLAS半导体公司新研制的DS1624型高分辨力智能温度传感器,能输出13位二进制数据,其分辨力高达0.03125°C,测温精度为±0.2°C。为了提高多通道智能温度传感器的转换速率,也有的芯片采用高速逐次逼近式A/D转换器。目前,智能温度传感器的总线技术也实现了标准化、规范化,所采用的总线主要有单线(1-Wire)总线、I2C总线、SMBus总线和spI总线。温度传感器作为从机可通过专用总线接口与主机进行通信。

方案一:

由于本设计是测温电路,可以使用热敏电阻之类的器件利用其感温效应,在将随被测温度变化 的电压或电流采集过来,进行 A/D 转换后,就可以用单片机进行数据的处理,在显示电路上,就可以将被测温度显示出来,这种设计需要用到 A/D 转换电路,感温电路比较麻烦。

方案二:

进而考虑到用温度传感器,在单片机电路设计中,大多都是使用传感器,所以这是非常容易想到的,所以可以采用一只温度传感器 DS18B20,此传感器,可以很容易直接读取被测温度值,进行转换,就可以满足设计要求。

从以上两种方案,两种都完全能够满足设计需要,很容易看出,采用方案二,电路比较简单,软件设计也比较简单,故采用了方案二。

1.2.3 掉电保持方案论证

为了方便使用,本设计采用串行E2PROMI2C-BUS的存储器件AT24C02,由于其具有接口方便,体积小,数据掉电不丢失等特点,在仪器仪表及工业自动化控制中得到大量的应用。本设计就是采用此功能。

1.3 电路设计最终方案决定

综上各方案所述,对此次作品的方案选定:采用STC89C51单片机作为主控制系统;采用DS18B20为传感器;采用AT24C02作为数据记录系统;采用数码管作为显示器件。

2 主要元件介绍

2.1 STC89C51介绍

STC89C51是由深圳宏晶科技公司生产的与工业标准MCS-51指令集和输出管脚相兼容的单片机。

2.1.1 STC89C51主要功能及PDIP封装

STC89C51主要功能如表1所示,其PDIP封装如图1所示

主要功能特性

兼容MCS51指令系统

8K可反复擦写Flash ROM

32个双向I/O口

256x8bit内部RAM

3个16位可编程定时/计数器中断

时钟频率0-24MHz

2个串行中断

可编程UART串行通道

2个外部中断源

共6个中断源

2个读写中断口线

3级加密位

低功耗空闲和掉电模式

软件设置睡眠和唤醒功能

表1:STC89C51主要功能

2.1.2 STC89C51引脚介绍

① 主电源引脚(2根)

VCC(Pin40):电源输入,接+5V电源

GND(Pin20):接地线

②外接晶振引脚(2根)

XTAL1(Pin19):片内振荡电路的输入端

XTAL2(Pin20):片内振荡电路的输出端

③控制引脚(4根)

RST/VPP(Pin9):复位引脚,引脚上出现2个机器周期的高电平将使单片机复位。

ALE/PROG(Pin30):地址锁存允许信号

PSEN(Pin29):外部存储器读选通信号

EA/VPP(Pin31):程序存储器的内外部选通,接低电平从外部程序存储器读指令,如果接高电平则从内部程序存储器读指令。

④可编程输入/输出引脚(32根)

STC89C51单片机有4组8位的可编程I/O口,分别位P0、P1、P2、P3口,每个口有8位(8根引脚),共32根。

P0口(Pin39~Pin32):8位双向I/O口线,名称为P0.0~P0.7

P1口(Pin1~Pin8):8位准双向I/O口线,名称为P1.0~P1.7

P2口(Pin21~Pin28):8位准双向I/O口线,名称为P2.0~P2.7

P3口(Pin10~Pin17):8位准双向I/O口线,名称为P3.0~P3.7

图1:STC89C51封装图

2.1.3 单片机最小系统:

当在STC89C51单片机的RST引脚引入高电平并保持2个机器周期时,单片机内部就执行复位操作,按键手动复位有电平方式和脉冲方式两种。其中电平复位是通过RST端经过电阻与电源VCC接通而实现的。最小系统如图2所示。

图2 单片机最小系统电路

电路以STC89C51单片机最小系统为控制核心,测温电路由DS18B20提供,输入部分采用三个独立式按键S1、S2、S3。数码管显示部分。具体电路连接,详见附录1。

2.2 DS18B20传感器介绍

2.2.1 DS18B20概述

在现代检测技术中,传感器占据着不可动摇的重要位置。主机对数据的处理能力已经相当的强,但是对现实世界中的模拟量却无能为力。如果没有各种精确可靠的传感器对非电量和模拟信号进行检测并提供可靠的数据,那计算机也无法发挥他应有的作用。传感器把非电量转换为电量,经过放大处理后,转换为数字量输入计算机,由计算机对信号进行分析处理。从而传感器技术与计算机技术结合起来,对自动化和信息化起重要作用。

采用各种传感器和微处理技术可以对各种工业参数及工业产品进行测控及检验,准确测量产品性能,及时发现隐患。为提高产品质量、改进产品性能,防止事故发生提供必要的信息和更可靠的数据。由于系统的工作环境比较恶劣,且对测量要求比较高,所以选择合适的传感器很重要。目前,国际上新型温度传感器正从模拟式向数字式、从集成化向智能化和网络化的方向飞速发展。智能温度传感器DS18B20正是朝着高精度、多功能、总线标准化、高可靠性及安全性、开发虚拟传感器和网络传感器、研制单片测温系统等高科技的方向迅速发展。因此,智能温度传感器DS18B20作为温度测量装置已广泛应用于人民的日常生活和工农业生产中。

美国DALLAS公司生产的 DS18B20可组网数字温度传感器芯片外加不锈钢保护管封装而成,具有耐磨耐碰,体积小,使用方便,封装形式多样,适用于各种狭小空间设备数字测温和控制领域。有独特的单线接口方式,DS1820在与微处理器连接时仅需要一条口线即可实现微处理器与DS1820的双向通讯;其测温范围 -55℃~+125℃,固有测温分辨率0.5℃;支持多点组网功能;多个DS1820可以并联在唯一的三线上,实现多点测温;工作电源为3~5V/DC;在使用中不需要任何外围元件。

DS18B20的性能特点如下:

(1) 采用DALLAS公司独特的单线接口方式:DS18B20与微处理器连接时仅需要一条口线即可实现微处理器与DS18B20的双向通讯;

(2)在使用中不需要任何外围元件;

(3)可用数据线供电,供电电压范围:+3.0V~+5.5V;

(4)测温范围:-55~+125℃。固有测温分辨率为0.5℃。当在-10℃~+85℃范围内,可确保测量误差不超过0.5℃,在-55~+125℃范围内,测量误差也不超过2℃;

(5)通过编程可实现9~12位的数字读数方式;

(6)用户可自设定非易失性的报警上下限值;

(7)支持多点的组网功能,多个DS18B20可以并联在唯一的三线上,实现多点测温

(8)负压特性,即具有电源反接保护电路。当电源电压的极性反接时,能保护DS18B20不会因发热而烧毁,但此时芯片无法正常工作;

(9)DS18B20的转换速率比较高,进行9位的温度值转换只需93.75ms;

(10)适配各种单片机或系统;

(11)内含64位激光修正的只读存储ROM,扣除8位产品系列号和8位循环冗余校验码(CRC)之后,产品序号占48位。出厂前产品序号存入其ROM中。在构成大型温控系统时,允许在单线总线上挂接多片DS18B20。

2.2.2 DS18B20引脚介绍

图3:DS18B20引脚

各引脚功能为:I/O为数据输入/输出端(即单线总线),它属于漏极开路输出,外接上拉电阻后,常态下呈高电平。UDD是可供选用的外部电源端,不用时接地,GND为地,NC空脚。

2.2.3 DS18B20的内部结构

DS18B20的内部结构主要包括7部分:寄生电源、温度传感器、64位激光(loser)ROM与单线接口、高速暂存器(即便筏式RAM,用于存放中间数据)、TH触发寄存器和TL触发寄存器,分别用来存储用户设定的温度上下限值、存储和控制逻辑、位循环冗余校验码(CRC)发生器。

图4:DS18B20内部结构

2.2.4 DS18B20的程序流程图

图5程序流程图

2.3 数码管介绍

数码管是一种半导体发光器件,其基本单元是发光二极管。数码管按段数分为七段数码管和八段数码管,八段数码管比七段数码管多一个发光二极管单元(多一个小数点显示);按能显示多少个“8”可分为1位、2位、4位等等数码管;

  按发光二极管单元连接方式分为共阳极数码管和共阴极数码管。共阳数码管是指将所有发光二极管的阳极接到一起形成公共阳极(COM)的数码管,共阳数码管在应用时应将公共极COM接到+5V,当某一字段发光二极管的阴极为低电平时,相应字段就点亮,当某一字段的阴极为高电平时,相应字段就不亮。共阴数码管是指将所有发光二极管的阴极接到一起形成公共阴极(COM)的数码管,共阴数码管在应用时应将公共极COM接到地线GND上,当某一字段发光二极管的阳极为高电平时,相应字段就点亮,当某一字段的阳极为低电平时,相应字段就不亮。

2.3.1 数码管概述

图5:数码管

数码显示器是一种由LED发光二极管组合显示字符的显示器件,它使用了8个Led发光二极管,其中七个用于显示字符,一个显示小数点,所以通称为七段发光二极管数码显示器。4位一体数码管,其内部段已连接好,引脚如图所示(数码管的正面朝自己,小数点在下方)。a、b、c、d、e、f、g、dp为段引脚,S1、S2、S3、S4分别表示四个数码管的位。

2.4 AT24C02简介

如图1为AT24C02的芯片引脚图。

图3-1  AT24C02的芯片引脚图

AT24C02提供电可擦除的串行1024位存储或可编程只读存储器(EEPROM)128字(8位/字)。

芯片在低压的工业与商业应用中进行了最优化。AT24C01的封装为8脚PDIP、8脚JEDEC

SOIC、8脚TSSOP,通过2线制串行接口进行数据传输。另外,整个系列有2.7V(2.7V至5.5V)和1.8V (1.8V至5.5V)两个版本。

设备操作:

C L O C K 和 D A T A 变化:SDA管脚通常外都要拉高。SDA管脚上的数据只能在SCL低期间改变。数据在SCL高期间改变定义为一个开始或停止信号。

开始状态:在任何操作之前必须有一个开始信号----在SCL为高时SDA上产生一个下降沿。

停止状态: SCL为高时SDA产生一个上升沿是停止信号,停止信号后将停止所有通信。

在一个读的序列之后,停止信号将让EEPROM进入备用电源模式。

2.4.1 I2C总线说明

I2C(Inter-Integrated Circuit)总线是一种由PHILIPS公司开发的两线式串行总线,用于连接微控制器及其外围设备。I2C总线产生于在80年代,最初为音频和视频设备开发,如今主要在服务器管理中使用,其中包括单个组件状态的通信。例如管理员可对各个组件进行查询,以管理系统的配置或掌握组件的功能状态,如电源和系统风扇。可随时监控内存、硬盘、网络、系统温度等多个参数,增加了系统的安全性,方便了管理。

1 I2C总线的硬件结构

I2C串行总线一般有两根信号线,一根是双向的数据线SDA,另一根是时钟线SCL。所有接到I2C总线设备上的串行数据SDA都接到总线的SDA上,各设备的时钟线SCL接到总线的SCL上。

为了避免总线信号的混乱,要求各设备连接到总线的输出端时必须是开漏输出或集电极开路输出。设备上的串行数据线SDA接口电路应该是双向的,输出电路用于向总线上发送数据,输入电路用于接收总线上的数据。而串行时钟线也应是双向的,作为控制总线数据传送的主机,一方面要通过SCL输出电路发送时钟信号,另一方面还要检测总线上的SCL电平,以决定什么时候发送下一个时钟脉冲电平;作为接受主机命令的从机,要按总线上的SCL信号发出或接收SDA上的信号,也可以向SCL线发出低电平信号以延长总线时钟信号周期。总线空闲时,因各设备都是开漏输出,上拉电阻RP使SDA和SCL线都保持高电平。任一设备输出的低电平都将使相应的总线信号线变低,也就是说:各设备的SDA是“与”关系,SCL也是“与”关系。

总线对设备接口电路的制造工艺和电平都没有特殊的要求(NMOS、CMOS都可以兼容)。在I2C总线上的数据传送率可高达每秒十万位,高速方式时在每秒四十万位以上。另外,总线上允许连接的设备数以其电容量不超过400pF为限。

总线的运行(数据传输)由主机控制。所谓主机是指启动数据的传送(发出启动信号)、发出时钟信号以及传送结束时发出停止信号的设备,通常主机都是微处理器。被主机寻访的设备称为从机。为了进行通讯,每个接到I2C总线的设备都有一个唯一的地址,以便于主机寻访。主机和从机的数据传送,可以由主机发送数据到从机,也可以由从机发到主机。凡是发送数据到总线的设备称为发送器,从总线上接收数据的设备被称为接受器。

I2C总线在传送数据过程中共有三种类型信号, 它们分别是:开始信号、结束信号和应答信号。

开始信号:SCL为高电平时,SDA由高电平向低电平跳变,开始传送数据。

结束信号:SCL为高电平时,SDA由低电平向高电平跳变,结束传送数据。

应答信号:接收数据的IC在接收到8bit数据后,向发送数据的IC发出特定的低电平脉冲,表示已收到数据。CPU向受控单元发出一个信号后,等待受控单元发出一个应答信号,CPU接收到应答信号后,根据实际情况作出是否继续传递信号的判断。若未收到应答信号,由判断为受控单元出现故障。如图3-2所示:

图3-2 开始、结束信号图

目前有很多半导体集成电路上都集成了I2C接口。带有I2C接口的单片机有:CYGNAL的 C8051F0XX系列,PHILIPSP87LPC7XX系列,MICROCHIP的PIC16C6XX系列等。很多外围器件如存储器、监控芯片等也提供I2C接口。

总线基本操作:

I2C规程运用主/从双向通讯。器件发送数据到总线上,则定义为发送器,器件接收数据则定义为接收器。主器件和从器件都可以工作于接收和发送状态。 总线必须由主器件(通常为微控制器)控制,主器件产生串行时钟(SCL)控制总线的传输方向,并产生起始和停止条件。SDA线上的数据状态仅在SCL为低电平的期间才能改变,SCL为高电平的期间,SDA状态的改变被用来表示起始和停止条件。

控制字节:在起始条件之后,必须是器件的控制字节,其中高四位为器件类型识别符(不同的芯片类型有不同的定义,EEPROM一般应为1010),接着三位为片选,最后一位为读写位,当为1时为读操作,为0时为写操作。

写操作:写操作分为字节写和页面写两种操作,对于页面写根据芯片的一次装载的字节不同有所不同。

读操作:读操作有三种基本操作:当前地址读、随机读和顺序读。图4给出的是顺序读的时序图。应当注意的是:最后一个读操作的第9个时钟周期不是“不关心”。为了结束读操作,主机必须在第9个周期时发出停止条件或者在第9个时钟周期内保持SDA为高电平、然后发出停止条件。

3 程序流程图



 

图6:程序流程图

结论

通过对自己在大学两年时间里所学的知识的回顾,并充分发挥对所学知识的理解和对毕业设计的思考及书面表达能力,最终完成了本设计。这为自己今后进一步深化学习,积累了一定宝贵的经验。撰写论文的过程也是专业知识的学习过程,它使我运用已有的专业基础知识,对其进行设计,分析和解决一个理论问题或实际问题,把知识转化为能力的实际训练。培养了我运用所学知识解决实际问题的能力。

本次论文设计,使我加深了对单片机的认识,并且熟悉了单片机系统的设计流程,收获丰硕。技术在不断进步,机械式时钟已经被淘汰,电子时代已经到来。做为新时代的我们,更应该提高自身能力,适应新时代的发展。知识来自实践,多从生活中探寻所需要的。

从这次的论文设计中,我真正的体会到,知识的重要性,特别是要理论联系实际,把我们所学的理论知识运用到实际生活当中,要用知识改变一切。

参考文献

[1] 陈权昌,李兴富.单片机原理及应用[M].广州:华南理工大学出版社,2007.84~102

[2] 李庆亮.C语言程序设计实用教程[M].北京:机械工业出版社,2005.32~58

[3] 杨志忠.数字电子技术[M].北京:高等教育出版社,2003.125~132

[4] 及力.Protel 99 SE原理图与PCB设计教程[M].北京:电子工业出版社,2007.89~150

[5] 徐江海.单片机实用教程[M].北京:机械工业出版社,2006.128~156

[6] 胡宴如.模拟电子技术[M].北京:高等教育出版社,2008.60~104

[7] 汪文,陈林.单片机原理及应用[M].湖北:华中科技大学出版社,2007.36~68

[8] 康华光.电子技术基础数字部分[M].北京:高等教育出版社,2008.203~209

[9] 杨欣.电子设计从零开始[M].北京:清华大学出版社,2005.28~102


致谢

首先,感谢学校三年来对我的培养。为我们营造了一个良好的学习氛围,建设一流的教学设施,使我们身心愉快的投入到学习中。

其次,感谢尊敬的指导老师,有了他的谆谆教诲,处处提点,才使本论文的前期准备以及整个研究过程顺利完成。指导老师的严谨治学态度、扎实的理论基础、全身心投入工作的精神以及对学生尽心尽力的态度给了我极大的帮助与鼓励,使我受益匪浅。从指导老师的教学态度上,我学到的不仅仅只有书本上的知识,还有做人的道理。他严肃的科学态度,严谨的治学精神,精益求精的工作作风,深深地感染和激励着我。在此谨向指导老师致以诚挚的谢意和崇高的敬意。

最后,感谢我的父母多年来给予我的支持和关怀,同时感谢我的舍友和朋友对我的帮助。


附录1 系统原理图


附录2  C语言程序


单片机源程序如下:

  1. #include<reg52.h>
  2. #define ui unsigned int
  3. #define uc unsigned char                 //宏定义
  4. sbit SET=P3^1;                                              //定义调整键
  5. sbit DEC=P3^2;                                              //定义减少键
  6. sbit ADD=P3^3;                                              //定义增加键
  7. sbit BEEP=P3^6;                                              //定义蜂鸣器
  8. sbit ALAM=P1^2;                                                        //定义灯光报警
  9. sbit ALAM1=P1^4;
  10. sbit DQ =P3^7;                                               //定义DS18B20总线I/O            
  11. sbit SCL=P1^6;
  12. sbit SDA=P1^7;
  13. sbit DIAN=P0^5;                                    //小数点
  14. bit bdata shanshuo_st;                                              //闪烁间隔标志
  15. bit bdata beep_st;                                                             //蜂鸣器间隔标志
  16. uc x=0;                                                              //计数器
  17. ui bai,shi,ge;
  18. uc set_st=0;                                               //状态标志
  19. char shangxian,xiaxian;
  20. code LEDData[]={0x5F,0x44,0x9D,0xD5,0xC6,0xD3,0xDB,0x47,0xDF,0xD7,0xCF,0xDA,0x9B,0xDC,0x9B,0x8B};
  21. //====================================DS18B20=========================================
  22. /*****延时子程序*****/
  23. void Delay_DS18B20(int num)
  24. {
  25.   while(num--) ;
  26. }
  27. void delay()//5微秒延时函数
  28. { ;; }
  29. void start()  //开始信号
  30. {            
  31.               SDA=1;
  32.               delay();
  33.               SCL=1;
  34.               delay();
  35.               SDA=0;
  36.               delay();
  37. }
  38. void stop()   //终止信号
  39. {
  40.               SDA=0;
  41.               delay();
  42.               SCL=1;
  43.               delay();
  44.               SDA=1;
  45.               delay();
  46. }
  47. void respons()  //应答
  48. {
  49.               uc i;
  50.               SCL=1;
  51.               delay();
  52.               while((SDA==1)&&(i<250))i++;//如果SDA为低应答有效,或者超过一定时间默认应答有效
  53.               SCL=0;
  54.               delay();
  55. }
  56. void init24c04()//I2C总线初始化
  57. {
  58.               SDA=1;
  59.               delay();
  60.               SCL=1;
  61.               delay();
  62. }
  63. void write_byte(uc date)//写操作
  64. {
  65.               uc i,temp;
  66.               temp=date;
  67.               for(i=0;i<8;i++)
  68.               {
  69.                             temp=temp<<1;
  70.                             SCL=0;
  71.                                delay();
  72.                             SDA=CY;
  73.                             delay();
  74.                             SCL=1;
  75.                             delay();
  76.               }
  77.               SCL=0;
  78.               delay();
  79.               SDA=1;
  80.               delay();
  81. }
  82. uc read_byte()//读操作
  83. {
  84.               uc i,k;
  85.               SCL=0;
  86.               delay();
  87.               SDA=1;
  88.               delay();
  89.               for(i=0;i<8;i++)
  90.               {
  91.                             SCL=1;
  92.                             delay();            
  93.                             k=(k<<1)|SDA;
  94.                             SCL=0;
  95.                             delay();            
  96.               }
  97.               return k;
  98. }
  99. void write_add(uc address,uc date)//往任意地址存数据
  100. {
  101.               start();
  102.               write_byte(0xa0);//0xa0代表写入
  103.               respons();
  104.               write_byte(address);
  105.               respons();
  106.               write_byte(date);
  107.               respons();
  108.               stop();
  109. }
  110. uc read_add(uc address)//读随意地址内容
  111. {
  112.               uc date;
  113.               start();
  114.               write_byte(0xa0);//0xa0代表写入
  115.               respons();
  116.               write_byte(address);
  117.               respons();
  118.               start();
  119.               write_byte(0xa1);//0xa1代表读出
  120.               respons();
  121.               date=read_byte();
  122.               stop();
  123.               return date;//返回值
  124. }
  125. /*****初始化DS18B20*****/
  126. void Init_DS18B20(void)
  127. {
  128.   uc w=0;
  129.   DQ = 1;         //DQ复位
  130.   Delay_DS18B20(8);    //稍做延时
  131.   DQ = 0;         //单片机将DQ拉低
  132.   Delay_DS18B20(80);   //精确延时,大于480us
  133.   DQ = 1;         //拉高总线
  134.   Delay_DS18B20(14);
  135.   w = DQ;           //稍做延时后,如果x=0则初始化成功,x=1则初始化失败
  136.   Delay_DS18B20(20);
  137. }
  138. /*****读一个字节*****/
  139. unsigned char ReadOneChar(void)
  140. {
  141.   uc i=0;
  142.   uc dat = 0;
  143.   for (i=8;i>0;i--)
  144.   {
  145.     DQ = 0;     // 给脉冲信号
  146.     dat>>=1;
  147.     DQ = 1;     // 给脉冲信号
  148.     if(DQ)
  149.     dat|=0x80;
  150.     Delay_DS18B20(4);
  151.   }
  152.   return(dat);
  153. }
  154. /*****写一个字节*****/
  155. void WriteOneChar(uc dat)
  156. {
  157.   uc i=0;
  158.   for (i=8; i>0; i--)
  159.   {
  160.     DQ = 0;
  161.     DQ = dat&0x01;
  162.     Delay_DS18B20(5);
  163.     DQ = 1;
  164.     dat>>=1;
  165.   }
  166. }
  167. /*****读取温度*****/
  168. ui ReadTemperature(void)
  169. {
  170.   ui b=0;
  171.   ui t=0;
  172.   ui a=0;
  173.   Init_DS18B20();
  174.   WriteOneChar(0xCC);  //跳过读序号列号的操作
  175.   WriteOneChar(0x44);  //启动温度转换
  176.   Init_DS18B20();
  177.   WriteOneChar(0xCC);  //跳过读序号列号的操作
  178.   WriteOneChar(0xBE);  //读取温度寄存器
  179.   a=ReadOneChar();     //读低8位
  180.   b=ReadOneChar();    //读高8位
  181.   t=b;
  182.   t<<=8;
  183.   t=t|a;
  184.   tt=t*0.0625;
  185.   t= tt*10+0.5;     //放大10倍输出并四舍五入
  186.   return(t);
  187. }
  188. /*****延时子程序*****/
  189. void Delay(ui num)
  190. {
  191. while( --num );
  192. }
  193. /*****初始化定时器0*****/
  194. void InitTimer(void)
  195. {
  196.               TMOD=0x01;
  197.               TH0=0x3c;
  198.               TL0=0xb0;     //50ms(晶振12M)
  199. }
  200. /*****读取温度*****/
  201. void check_wendu(void)
  202. {
  203.               ui f;
  204.               f=ReadTemperature()-5;                                            //获取温度值并减去DS18B20的温漂误差
  205.               if(f<0)f=0;
  206.               if(f>999)f=999;
  207.               bai=f/100;                                                                                         //计算得到十位数字
  208.               shi=(f%100)/10;                                                                          //计算得到个位数字
  209.               ge=(f%100)%10;                                                                                          //计算得到整数位
  210. }
  211. /*****显示开机初始化等待画面*****/
  212. void Disp_init(void)   
  213. {
  214.               P0 = ~0x80;      //显示----
  215.               P2 = 0x7F;
  216.               Delay(200);
  217.               P2 = 0xDF;
  218.               Delay(200);  
  219.               P2 = 0xF7;
  220.               Delay(200);
  221.               P2 = 0xFD;
  222.               Delay(200);
  223.               P2 = 0xFF;         //关闭显示
  224. }
  225. /*****显示温度子程序*****/
  226. void Disp_Temperature(void)     //显示温度
  227. {
  228.               P0 = ~0x98;      //显示C
  229.               P2 = 0x7F;
  230.               Delay(400);
  231.               P0=~LEDData[ge];    //显示个位
  232.               P2 = 0xDF;
  233.               Delay(400);
  234.               P0 =~LEDData[shi];    //显示十位
  235.               DIAN = 0;         //显示小数点
  236.               P2 = 0xF7;
  237.               Delay(400);
  238.               P0 =~LEDData[bai];    //显示百位
  239.               P2 = 0xFD;
  240.               Delay(400);
  241.               P2 = 0xff;         //关闭显示
  242. }
  243. /*****显示报警温度子程序*****/
  244. void Disp_alarm(uc baojing)
  245. {
  246.               P0 =~0x98;      //显示C
  247.               Delay(200);
  248.               P0 =~LEDData[baojing%10]; //显示十位
  249.               P2 = 0xDF;
  250.               Delay(200);
  251.               P0 =~LEDData[baojing/10]; //显示百位
  252.               P2 = 0xF7;
  253.               Delay(200);
  254.               if(set_st==1)P0 =~0xCE;
  255.               else if(set_st==2)P0 =~0x1A; //上限H、下限L标示
  256.               P2 = 0xFD;
  257.               Delay(200);
  258.               P2 = 0xff;         //关闭显示
  259. }
  260. /*****报警子程序*****/
  261. void Alarm()
  262. {
  263.               if(x>=10){beep_st=~beep_st;x=0;}
  264.               if((bai*10+shi)>=shangxian&&beep_st==1)
  265.               {
  266.                             BEEP=0;
  267.                             ALAM1=0;
  268.               }
  269.               else if((bai*10+shi)>=shangxian&&beep_st==0)
  270.               {
  271.                             BEEP=1;
  272.                             ALAM1=0;
  273.               }
  274.               if((bai*10+shi)<xiaxian&&beep_st==1)
  275.               {
  276.                             BEEP=0;
  277.                             ALAM=0;
  278.               }
  279.               else if((bai*10+shi)<xiaxian&&beep_st==0)
  280.               {
  281.                             BEEP=1;
  282.                             ALAM=0;
  283.               }
  284.               if(((bai*10+shi)<shangxian)&&((bai*10+shi)>=xiaxian))
  285.               {
  286.                             BEEP=1;
  287.                             ALAM1=1;
  288.                             ALAM=1;
  289.               }
  290. }
  291. /*****主函数*****/
  292. void main(void)
  293. {
  294.               ui z;
  295.               InitTimer();    //初始化定时器
  296.               EA=1;      //全局中断开关
  297.               ET0=1;      //开启定时器0
  298.               check_wendu();
  299.               check_wendu();
  300.               shangxian=read_add(10);
  301.               xiaxian=read_add(20);
  302.               for(z=0;z<300;z++)
  303.               {
  304.                             Disp_init();      
  305.               }
  306.               while(1)
  307.               {
  308.                             if(SET==0)
  309.                             {
  310.                                           Delay(2000);
  311.                                           do{}
  312.                                           while(SET==0);
  313.                                           set_st++;x=0;shanshuo_st=1;
  314.                                           if(set_st>2)set_st=0;
  315.                             }
  316.                             if(set_st==0)
  317.                             {
  318.                                           check_wendu();
  319.                                           Disp_Temperature();
  320.                                           Alarm();   //报警检测
  321.                             }
  322.                             else if(set_st==1)
  323.                             {
  324.                                           BEEP=1;    //关闭蜂鸣器
  325.                                           ALAM=1;
  326.                                           ALAM1=1;
  327.                                           if(x>=10){shanshuo_st=~shanshuo_st;x=0;}
  328.                                           if(shanshuo_st) {Disp_alarm(shangxian);}
  329.                                           if(ADD==0)
  330.                                           {
  331.                                                         do{Disp_alarm(shangxian);}
  332.                                                         while(ADD==0);
  333.                                                         shangxian++;
  334.                                                         if(shangxian>99)shangxian=99;
  335.                                                         write_add(10,shangxian);
  336.                                           }
  337.                                           else if(DEC==0)
  338. ……………………

     

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/658580.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

生产者延迟消息和重试机制

messageDelayLevel1s 5s 10s 30s 1m 2m 3m 4m 5m 6m 7m 8m 9m 10m 20m 30m 1h 2h 延迟消息级别public PutMessageResult putMessage(final MessageExtBrokerInner msg) {//事务消息处理if (tranType MessageSysFlag.TRANSACTION_NOT_TYPE|| tranType MessageSysFlag.TRANSACT…

静态测试---基于WorkList的活跃变量分析

本文主要用于记录在活跃变量分析实验中的报错及解决&#xff0c;涉及静态测试的详细原理内容较少&#xff0c;编译运行底层逻辑偏多。 一、实验要求 1&#xff09;使用llvm基于框架实现一个基于WorkList的活跃变量分析demo。变量在某个程序点有两种状态&#xff0c;live 或 dea…

如何用俄语预定酒店,柯桥外贸俄语培训

-Привет, Алекс! Как твои дела? 你好&#xff0c;阿列克斯&#xff01;你最近好吗&#xff1f; -Отлично! Я скоро уезжаю на 10 дней в Санкт-Петербург, но никак не могу найти там…

基于高光谱数据集的创新点实现-高斯核函数卷积神经网络

一、高光谱数据集简介 1.1 数据集简介 数据集链接在这:高光谱数据集(.mat.csv)-科研学术 数据集包含下面三个文件&#xff1a; 文件中包含.mat与.csv,145x145x220, 其实主要使用avirissub.csv文件&#xff0c;在代码上只是将mat文件转成了csv文件。具体avirissub.csv如下&am…

第二十四章多栏布局解决方案(什么是自适应?/)

什么是自适应? 指能使网页自适应显示在不同大小终端设备上新网页设计方式及技术.简单的来说自适应就是让同一个页面自动适应不同大小的设备&#xff0c;从而解决为不同设备提供不同版本的页面问题。 1&#xff0e;两列自适应 两列自适应布局是指左侧固定宽度&#xff0c;右…

小程序使用Canvas设置文字竖向排列

在需要使用的js页面引入js文件,传入对应参数即可 /** * 文本竖向排列 */ function drawTextVertical(context, text, x, y) {var arrText text.split();var arrWidth arrText.map(function (letter) {return 26;});var align context.textAlign;var baseline context.…

河北奥润顺达集团研究院PMO经理常江南受邀为第十三届中国PMO大会演讲嘉宾

全国PMO专业人士年度盛会 河北奥润顺达集团研究院PMO经理、研发部运营管理办负责人常江南先生受邀为PMO评论主办的2024第十三届中国PMO大会演讲嘉宾&#xff0c;演讲议题为“初建PMO的体系宣贯和人员培养实践总结”。大会将于6月29-30日在北京举办&#xff0c;敬请关注&#xf…

质量工具系列之Dependency-Track

项目开发中依赖了很多第三方开源工具&#xff0c;对于其版本&#xff0c;漏洞等因为时间或者是数量太多而无法关注到&#xff0c;Dependency-Track解决这些问题。 Dependency-Track 是一个开源组件分析平台&#xff0c;是开放网络应用安全项目&#xff08;OWASP&#xff09;的一…

【自动化运营】PlugLink 1.0开源版发布

什么是PlugLink&#xff1f; PlugLink&#xff0c;顾名思义&#xff0c;就是插件的链接。它旨在帮助个人和小微企业实现运营自动化&#xff0c;通过链接脚本、API、AI大模型等&#xff0c;实现全自动工作流程。你可以把PlugLink看作一个巨大的拼装积木&#xff0c;每一个插件都…

Java解析JSON并修改属性值:从JSON到JsonObject的逐层解析

哈喽&#xff0c;大家好&#xff0c;我是木头左&#xff01; 在Java中&#xff0c;可以使用各种库来解析和操作JSON数据。其中&#xff0c;Gson和Jackson是两个非常流行且功能强大的库。在这篇文章中&#xff0c;将使用Gson库来解析给定的JSON字符串&#xff0c;修改operationB…

加氢站压缩液驱比例泵放大器

加氢站压缩液驱液压系统的要求是实现换向和速度控制&#xff0c;对液压动力机构而言&#xff0c;按原理可区分为开式&#xff08;阀控&#xff09;- 节流控制系统和闭式&#xff08;泵控&#xff09;- 容积控制系统&#xff1a; 阀控系统 – 节流调速系统&#xff1a;由BEUEC比…

Sui新共识协议刷新了区块链交易速度的标准

Sui是提供业界领先性能和无限水平扩展的创新Layer 1区块链&#xff0c;今日在官推上宣布其最新共识协议Mysticeti已成功部署到测试网。这一重大突破将Sui测试网的共识时间减少了80%&#xff0c;至390毫秒&#xff0c;同时保持协议的行业领先吞吐量。这一令人印象深刻的演示证明…

英国金融时报:波场TRON出席康奈尔大学区块链会议

近日,康奈尔区块链大会的白金赞助商波场TRON在罗斯福岛的康奈尔大学科技校区(Cornell Tech)举办多项活动,消息得到英国金融时报,费加罗报和Benzinga等权威外媒报道,这全面彰显了波场TRON的领导力。大会吸引了包括学生、学者和行业领袖等在内的 800 多名参与者,凸显了波场TRON致…

【教程】Linux部署Android安卓模拟器

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhagn.cn] 如果本文帮助到了你&#xff0c;欢迎[点赞、收藏、关注]哦~ 未完成&#xff0c; 先简单记录下指令。 docker-android https://github.com/budtmo/docker-android 检查系统是否支持&#xff1a; sudo apt instal…

【ARM+Codesys案例】树莓派+Codesys软PLC方案在包装行业灌装旋盖机的应用

ARM系列支持&#xff1a;全志T3、RK3568、树莓派 机型定义&#xff1a;双工位旋盖机 旋盖机主要适用于不同规格的材质及不同规格的盖、旋&#xff08;轧&#xff09;盖。适用螺旋盖、防盗盖、防撞盖、压入盖等。压力可方便调整&#xff0c;根据瓶盖大小设置取盖位。结构紧凑、…

【Springboot】注解@JsonIgnore、@TableField、@TableId

文章目录 1.JsonIgnore2.TableField(exist false)3.TableId 1.JsonIgnore 2.TableField(exist false) 新增一个字符串字段&#xff0c;并且这个字段是不用到数据库查询的 3.TableId Mybatis-plus 出现 WHERE null?&#xff0c;此时就要加这个注解。

全球十大体育赛事API服务

体育赛事API汇总&#xff1a; Broadage全球橄榄球赛事数据Broadage全球棒球赛事数据Broadage全球篮球实时数据Broadage全球冰球赛事数据Broadage全球排球实时数据TennisApi全球网球赛事讯息Broadage全球足球实时数据棒球数据【纳米数据】

AACR美国癌症研究学会文献及补充文献下载的途径

美国癌症研究学会American Association for Cancer Research&#xff08;AACR&#xff09;创建于1907年&#xff0c;是世界上成立最早、规模最大的致力于全面、创新和高水准癌症研究的科学组织。其出版物包括7种正式出版的期刊&#xff08;其中五种经同行评议*&#xff09;和其…

C++重点基础知识汇总大全

文章目录 一些基础知识点指针和引用 一些基础知识点 1、十进制的数字比较长的时候&#xff0c;可以加方便阅读到底是几位&#xff0c;输出的时候跟不加是一样的效果 // 十进制可以加 cout << 13890324 << endl; // 13890324 // 二进制前加0b cout << 0b111…

Web前端复习二

第一章测试 选项中加粗的为答案 1.图片的边框可以通过( )设定宽度。 A.width B.height C.border D.align 2.关于超链接&#xff0c;( )属性用于规定在何处打开链接文档。 A.href . B.target C.title D.onclick 3.( )是在新窗口打开网页文档。 A _blank B_self C_…