如何用单片机和DS18B20进行多点温度测试?(DS18B20的序列号怎么读取?)
请教高手一下!我做的是一个单片机和多个DS18B20测温,但是我不知道怎么读出DS18B20的序列号。请各位指点一下怎么才能用一个单片机测试多点的温度?谢谢各位了!呵呵。我很着急啊!请多多帮忙啊! 写指令0X33,然后从18B20读出的8个字节就是序列号。(总线上只有一个18B20)写指令0X55,然后再写8个字节的序列号,再下来的操作就只是针对该序列号的18B20 谢谢你的指点啊!我做的是一个仿真,DS18B20的序列号怎么读啊?它的序列号是否可以在其属性中设定呢?(在Proteous软件仿真中) 楼主的意思是总线上带多个18B20的时候吧,你可以去DALLAS网站去下载学习他们给出的程序,如何搜索单总线上挂多个1820的方法,不过他们写的比较烦琐,看的人头比较疼 郁闷啊!谁有简单的啊?参考一下。 美信网站上的应用笔记已经非常简单明确了,从算法到实例程序都有,楼主还需要怎样的【参考】呢? 呵呵,不好意思啊!新手哦!以后还要大家多多帮忙啊!先谢谢各位单片机“前辈”了! 我找了点资料,可都是很复杂啊!看了两天,头都大了。难道搞编程就是这么难吗? 美信的不要看,文档简直是太次了写的,我给你个达拉斯写的搜索吧
点击此处下载 ourdev_291292.pdf(文件大小:194K) (原文件名:18b20搜索方法.pdf) 谢谢啊!不过毕业前恐怕做不好了。等毕业后继续做吧。 当年我参加电子设计竞赛的时候也没能取出序列号
后面用了个笨办法,你不是单总线吗?好,我就让你一个18b20一条单总线!~还好当时要的传感器不多 楼上的好夸张
是在不行的话,好像有一个 转接芯片 专门从1-Wire协议转到I2C协议的
整个这玩意,应该就容易得多了吧 哦
谢谢楼上的,下次再做的话试试 /
美信的不要看,文档简直是太次了写的,我给你个达拉斯写的搜索吧
点击此处下载 ourdev_291292.pdf(文件大小:194K) (原文件名:18b20搜索方法.pdf)
/
PS:达拉斯就是美信的全资子公司吧~ 写指令0X33,然后从18B20读出的8个字节就是序列号。(总线上只有一个18B20) 。依次挂 接多个18B20,分别读出序列号。将读出的序列号放在一个数组里面,分别操作就可以 14 楼太对拉. 贴个参考的程序, 有啥问题自己再调调哈,是LCD1602显示的.
#include <reg51.h>
#include <intrins.h>
#define uchar unsigned char
#define uintunsigned int
sbit DQ = P3^3;//定义DS18B20端口DQ
sbit BEEP=P3^7 ; //蜂鸣器驱动线
bitpresence ;
sbit LCD_RS = P2^0;
sbit LCD_RW = P2^1;
sbit LCD_EN = P2^2;
uchar codecdis1[ ] = {" DS18B20 OK "};
uchar codecdis2[ ] = {" "};
uchar codecdis3[ ] = {" DS18B20ERR0R "};
uchar codecdis4[ ] = {"PLEASE CHECK"};
unsigned char datadisplay = {0x00,0x00};
unsigned char dataRomCode = {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00};
unsigned char Temp;
unsigned charcrc;
void beep();
#define delayNOP(); {_nop_();_nop_();_nop_();_nop_();};
/*******************************************************************/
void delay1(int ms)
{
unsigned char y;
while(ms--)
{
for(y = 0; y<250; y++)
{
_nop_();
_nop_();
_nop_();
_nop_();
}
}
}
/******************************************************************/
/* */
/*检查LCD忙状态 */
/*lcd_busy为1时,忙,等待。lcd-busy为0时,闲,可写指令与数据。 */
/* */
/******************************************************************/
bit lcd_busy()
{
bit result;
LCD_RS = 0;
LCD_RW = 1;
LCD_EN = 1;
delayNOP();
result = (bit)(P0&0x80);
LCD_EN = 0;
return(result);
}
/*******************************************************************/
/* */
/*写指令数据到LCD */
/*RS=L,RW=L,E=高脉冲,D0-D7=指令码。 */
/* */
/*******************************************************************/
void lcd_wcmd(uchar cmd)
{
while(lcd_busy());
LCD_RS = 0;
LCD_RW = 0;
LCD_EN = 0;
_nop_();
_nop_();
P0 = cmd;
delayNOP();
LCD_EN = 1;
delayNOP();
LCD_EN = 0;
}
/*******************************************************************/
/* */
/*写显示数据到LCD */
/*RS=H,RW=L,E=高脉冲,D0-D7=数据。 */
/* */
/*******************************************************************/
void lcd_wdat(uchar dat)
{
while(lcd_busy());
LCD_RS = 1;
LCD_RW = 0;
LCD_EN = 0;
P0 = dat;
delayNOP();
LCD_EN = 1;
delayNOP();
LCD_EN = 0;
}
/*******************************************************************/
/* */
/*LCD初始化设定 */
/* */
/*******************************************************************/
void lcd_init()
{
delay1(15);
lcd_wcmd(0x01); //清除LCD的显示内容
lcd_wcmd(0x38); //16*2显示,5*7点阵,8位数据
delay1(5);
lcd_wcmd(0x38);
delay1(5);
lcd_wcmd(0x38);
delay1(5);
lcd_wcmd(0x0c); //显示开,关光标
delay1(5);
lcd_wcmd(0x06); //移动光标
delay1(5);
lcd_wcmd(0x01); //清除LCD的显示内容
delay1(5);
}
/*******************************************************************/
/* */
/*设定显示位置 */
/* */
/*******************************************************************/
void lcd_pos(uchar pos)
{
lcd_wcmd(pos | 0x80);//数据指针=80+地址变量
}
/*******************************************************************/
/* */
/*us级延时函数 */
/* */
/*******************************************************************/
void Delay(unsigned int num)
{
while( --num );
}
/*******************************************************************/
/* */
/*初始化ds1820 */
/* */
/*******************************************************************/
Init_DS18B20(void)
{
DQ = 1; //DQ复位
Delay(8); //稍做延时
DQ = 0; //将DQ拉低
Delay(90); //精确延时 大于 480us
DQ = 1; //拉高总线
Delay(8);
presence = DQ; //读取存在信号
Delay(100);
DQ = 1;
return(presence); //返回信号,0=presence,1= no presence
}
/*******************************************************************/
/* */
/* 读一位(bit) */
/* */
/*******************************************************************/
uchar read_bit(void)
{
unsigned char i;
DQ = 0; //将DQ 拉低开始读时间隙
DQ = 1; // then return high
for (i=0; i<3; i++); // 延时15μs
return(DQ); // 返回 DQ 线上的电平值
}
/*******************************************************************/
/* */
/* 读一个字节 */
/* */
/*******************************************************************/
ReadOneChar(void)
{
unsigned char i = 0;
unsigned char dat = 0;
//for (i = 8; i > 0; i--)
//{
// read_bit();
// DQ = 0; // 给脉冲信号
// dat >>= 1;
// DQ = 1; // 给脉冲信号
for (i=0;i<8;i++)
{ // 读取字节,每次读取一个字节
if(read_bit()) dat|=0x01<<i; // 然后将其左移
// if(DQ)
// dat |= 0x80;
Delay(4);
}
return (dat);
}
/*******************************************************************/
/* */
/* 写一位 */
/* */
/*******************************************************************/
void write_bit(char bitval) {
DQ = 0; // 将DQ 拉低开始写时间隙
if(bitval==1) DQ =1; // 如果写1,DQ 返回高电平
Delay(5); // 在时间隙内保持电平值,
DQ = 1; // Delay函数每次循环延时16μs,因此delay(5) = 104μs
}
/*******************************************************************/
/* */
/* 写一个字节 */
/* */
/*******************************************************************/
WriteOneChar(unsigned char dat)
{
unsigned char i = 0;
unsigned char temp;
//for (i = 8; i > 0; i--)
//{
for (i=0; i<8; i++) // 写入字节, 每次写入一位
{
// DQ = 0;
// DQ = dat&0x01;
// Delay(5);
// DQ = 1;
temp = dat>>i;
temp &= 0x01;
write_bit(temp);
// dat>>=1;
}
Delay(5);
}
/*******************************************************************/
/* */
/* 读取64位序列码 */
/* */
/*******************************************************************/
Read_RomCord(void)
{
unsigned char j;
Init_DS18B20();
WriteOneChar(0x33);// 读序列码的操作
for (j = 0; j < 8; j++)
{
RomCode = ReadOneChar() ;
}
}
/*******************************************************************/
/* */
/*DS18B20的CRC8校验程序 */
/* */
/*******************************************************************/
uchar CRC8()
{
uchar i,x; uchar crcbuff;
crc=0;
for(x = 0; x <8; x++)
{
crcbuff=RomCode;
for(i = 0; i < 8; i++)
{
if(((crc ^ crcbuff)&0x01)==0)
crc >>= 1;
else {
crc ^= 0x18; //CRC=X8+X5+X4+1
crc >>= 1;
crc |= 0x80;
}
crcbuff >>= 1;
}
}
return crc;
}
/*******************************************************************/
/* */
/* 数据转换与显示 */
/* */
/*******************************************************************/
Disp_RomCode()
{
uchar j;
uchar H_num=0x40; //LCD第二行初始位置
for(j=0;j<8;j++)
{
Temp = RomCode;
display=((Temp&0xf0)>>4);
if(display>9)
{ display=display+0x37;}
else{display=display+0x30;}
lcd_pos(H_num);
lcd_wdat(display); //高位数显示
H_num++;
display=(Temp&0x0f);
if(display>9)
{display=display+0x37;}
else {display=display+0x30;}
lcd_pos(H_num);
lcd_wdat(display); //低位数显示
H_num++;
}
}
/*******************************************************************/
/* */
/* 蜂鸣器响一声 */
/* */
/*******************************************************************/
void beep()
{
unsigned char y;
for (y=0;y<100;y++)
{
Delay(60);
BEEP=!BEEP; //BEEP取反
}
BEEP=1; //关闭蜂鸣器
Delay(40000);
}
/*******************************************************************/
/* */
/* DS18B20 OK 显示菜单 */
/* */
/*******************************************************************/
voidOk_Menu ()
{
ucharm;
lcd_init(); //初始化LCD
lcd_pos(0); //设置显示位置为第一行的第1个字符
m = 0;
while(cdis1 != '\0')
{ //显示字符
lcd_wdat(cdis1);
m++;
}
lcd_pos(0x40); //设置显示位置为第二行第1个字符
m = 0;
while(cdis2 != '\0')
{
lcd_wdat(cdis2); //显示字符
m++;
}
}
/*******************************************************************/
/* */
/* DS18B20 ERROR 显示菜单 */
/* */
/*******************************************************************/
voidError_Menu ()
{
ucharm;
lcd_init(); //初始化LCD
lcd_pos(0); //设置显示位置为第一行的第1个字符
m = 0;
while(cdis3 != '\0')
{ //显示字符
lcd_wdat(cdis3);
m++;
}
lcd_pos(0x40); //设置显示位置为第二行第1个字符
m = 0;
while(cdis4 != '\0')
{
lcd_wdat(cdis4); //显示字符
m++;
}
}
/*******************************************************************/
/* */
/* 主函数 */
/* */
/*******************************************************************/
void main()
{
P0 = 0xff;
P2 = 0xff;
while(1)
{
Ok_Menu ();
Read_RomCord(); //读取64位序列码
CRC8(); //CRC效验
if(crc==0) //CRC效验正确
{
Disp_RomCode(); //显示64位序列码
beep();
}
while(!presence)
{
Init_DS18B20();
delay1(1000);
}
Error_Menu ();
do
{
Init_DS18B20();
beep();
}
while(presence);
}
}
/*******************************************************************/ MARK mark MARK mark 太好啦~
· mark mark mark mark mark yanxinboy 发表于 2009-9-1 10:09 static/image/common/back.gif
14 楼太对拉. 贴个参考的程序, 有啥问题自己再调调哈,是LCD1602显示的.
#include
请问yanxinboy,多路的做过没,求讨论!卡在MATCH_ROM了...... 太好了,正需要这些资料,在此谢谢了! 路过学习一下!!!!!! mark、、 本帖最后由 xuzixin 于 2012-9-26 11:22 编辑
请看http://www.amobbs.com/forum.php?mod=viewthread&tid=5495639&page=1#pid6095341
很完整的 “本人项目用18B20测温带shell程序” mark…… 爱啃苹果的葱 发表于 2012-9-27 23:56 static/image/common/back.gif
mark……
我擦 在这碰见你! 呃呃呃呃!感谢了! mark、、下次用 楼主,我当时就很笨,一根线上就只有一个DS18B20,对每个都读取一下,然后 对每个编号,查表查出号码然后一个个读取 其实退出程序的还是蛮给力的、{:smile:} DS18B20封装上好像不标序列号的,那么我在想:总线上挂多个DS18B20时,即使我写一个程序把它们的序列号都读出来了,但并不能与实际的器件位置相对应啊,这又有什么用呢? nice~~~~~~
页:
[1]