wangnengjie 发表于 2014-9-11 11:35:36

多DS18S20温度传感器运行时间长的问题。

本人用4个传感器并联,放在屋顶上测温度。时间长了(一天到三天)经常读传感器出错。温度变成120+的数值。出问题的传感器随机。

环境大致是这样的,
传感器的线长大约20米。其中有5米和220V的电平行走的。
arduino带三线驱动的,上拉电阻开始用的100欧姆的,后来觉得有点小了换了1K的(4.7K的话就读不到温度了)。
arduino的循环读数据,具体循环的快慢取决于程序多少。没具体测试过,估计每秒运行几十次的循环是有的。

现在我在循环里加了延时语句,每次循环等5秒。看起来情况好不少了。但是不知道怎么彻底解决此问题。
用的Dalla,onewire的库。

hp198969 发表于 2014-9-11 14:05:10

ds18s20好像是每秒更新一次温度吧……你太频繁的操作,线又这么长,容易出问题啦……

wangnengjie 发表于 2014-9-11 14:22:40

线的质量还是不错的,我也担心是太频繁操作的原因。只是想搞清楚这种稳定性的问题根源在哪里。
目前加了延时用了整一天没出问题。再持续用一段时间看看。

baoz 发表于 2014-9-12 00:10:03

搭车请教下LZ,DS18S20并联,读数的时候是什么样的反馈?有办法识别设备位置么?

wangnengjie 发表于 2014-9-12 11:05:53

读数有冲突。但是有冲突解决办法的。Dalla那个函数库里有。请见我的源代码。

#include <OneWire.h>
#include <DallasTemperature.h>
#include <SPI.h>
#include <LCD5110_CN.h>
LCD5110 myGLCD(3,4,5,11,6);
extern uint8_t SmallFont[];
// Data wire is plugged into port 2 on the Arduino
#define ONE_WIRE_BUS 2
#define TEMPERATURE_PRECISION 12
#define RELAY_pump7
#define RELAY_spray18
#define RELAY_spray29
#define RELAY_spray310
#define OPEN_TEMPERATURE 45
#define CLOSE_TEMPERATURE 40
/** Pins:
* Hardware SPI:
* MISO -> 12
* MOSI -> 11
* SCK -> 13
*
* Configurable:
* CE -> 8
* CSN -> 7 */
// Setup a oneWire instance to communicate with any OneWire devices (not just Maxim/Dallas temperature ICs)

OneWire oneWire(ONE_WIRE_BUS);

// Pass our oneWire reference to Dallas Temperature.
DallasTemperature sensors(&oneWire);

int numberOfDevices; // Number of temperature devices found

DeviceAddress tempDeviceAddress; // We'll use this variable to store a found device address

// function to print a device address
void printAddress(DeviceAddress deviceAddress)
{
for (uint8_t i = 0; i < 8; i++)
{
    if (deviceAddress < 16) Serial.print("0");
    Serial.print(deviceAddress, HEX);
}
}

// function to print the temperature for a device
void printTemperature(DeviceAddress deviceAddress)
{
// method 1 - slower
//Serial.print("Temp C: ");
//Serial.print(sensors.getTempC(deviceAddress));
//Serial.print(" Temp F: ");
//Serial.print(sensors.getTempF(deviceAddress)); // Makes a second call to getTempC and then converts to Fahrenheit

// method 2 - faster
float tempC = sensors.getTempC(deviceAddress);
Serial.print("Temp C: ");
Serial.print(tempC);
//Serial.print(" Temp F: ");
//Serial.println(DallasTemperature::toFahrenheit(tempC)); // Converts tempC to Fahrenheit
}
float tempC;
char tempstr;
char ifspray=0;
float averagetemp;
int startnum=0;

void setup(void)
{
// start serial port
Serial.begin(9600);
   /* To change CE / CSN Pins:
   *
   * Mirf.csnPin = 9;
   * Mirf.cePin = 7;
   */
myGLCD. InitLCD();
myGLCD.setFont(SmallFont);
pinMode(RELAY_pump, OUTPUT);
pinMode(RELAY_spray1, OUTPUT);
pinMode(RELAY_spray2, OUTPUT);
pinMode(RELAY_spray3, OUTPUT);

//Serial.println("Dallas Temperature IC Control Library Demo");
//Mirf.spi = &MirfHardwareSpi;
//Mirf.init();
// Start up the library
//Mirf.setRADDR((byte *)"Tempstation");
// Grab a count of devices on the wire
//Mirf.payload = sizeof(unsigned long);
// locate devices on the bus
Serial.print("Locating devices...");
/*
   * To change channel:
   *
   * Mirf.channel = 10;
   *
   * NB: Make sure channel is legal in your area.
   */
//Mirf.config();
sensors.begin();
numberOfDevices = sensors.getDeviceCount();
Serial.print("Found ");
Serial.print(numberOfDevices, DEC);
Serial.println(" devices.");

// report parasite power requirements
Serial.print("Parasite power is: ");
if (sensors.isParasitePowerMode()) Serial.println("ON");
else Serial.println("OFF");

// Loop through each device, print out address
for(int i=0;i<numberOfDevices; i++)
{
    // Search the wire for address
    if(sensors.getAddress(tempDeviceAddress, i))
      {
                Serial.print("Found device ");
                Serial.print(i, DEC);
                Serial.print(" with address: ");
                printAddress(tempDeviceAddress);
                Serial.println();

                //Serial.print("Setting resolution to ");
                //Serial.println(TEMPERATURE_PRECISION,DEC);

                // set the resolution to 9 bit (Each Dallas/Maxim device is capable of several different resolutions)
                sensors.setResolution(tempDeviceAddress, TEMPERATURE_PRECISION);

                Serial.print("Resolution actually set to: ");
                Serial.print(sensors.getResolution(tempDeviceAddress), DEC);
                Serial.println();
      }else{
                Serial.print("Found ghost device at ");
                Serial.print(i, DEC);
                Serial.print(" but could not detect address. Check power and cabling");
      }
}

}


void loop(void)
{
ifspray=0;
averagetemp=0;
//unsigned long time = millis();
// call sensors.requestTemperatures() to issue a global temperature
// request to all devices on the bus
Serial.print("Requesting temperatures...");
sensors.requestTemperatures(); // Send the command to get temperatures
Serial.println("DONE");
myGLCD.print("sensor1:",LEFT,0);
myGLCD.print("sensor2:",LEFT,8);
myGLCD.print("sensor3:",LEFT,16);
myGLCD.print("sensor4:",LEFT,24);
myGLCD.print("Start count:",LEFT,32);
myGLCD.print("Average:",LEFT,40);

// Loop through each device, print out temperature data
for(int i=0;i<numberOfDevices; i++)
{
    // Search the wire for address
    if(sensors.getAddress(tempDeviceAddress, i))
      {
                // Output the device ID
                //Serial.print("Temperature for device: ");
                //Serial.println(i,DEC);

                // It responds almost immediately. Let's print out the data
                printTemperature(tempDeviceAddress); // Use a simple function to print out the data
                tempC = sensors.getTempC(tempDeviceAddress);
                myGLCD.printNumF(tempC,2,RIGHT,i*8);
                //Mirf.setTADDR((byte *)"Transfer");
                //Mirf.send((byte *)"Temp C: \n");
                Serial.print("Temp C: ");
                //Mirf.send((byte *) dtostrf(tempC,5,2,tempstr));
                Serial.println(tempC);
                //while(Mirf.isSending()){
                //}
                //delay(10);
                /*while(!Mirf.dataReady()){
                  //Serial.println("Waiting");
                  if ( ( millis() - time ) > 50000 ) {
                     Serial.println("Timeout on response from server!");
                     return;
                  }
                }*/
               
      //else ghost device! Check your power requirements and cabling
      }
      averagetemp=averagetemp+tempC;
      
}
Serial.println(numberOfDevices);
averagetemp=averagetemp/numberOfDevices;
myGLCD.printNumF(averagetemp,2,RIGHT,40);
myGLCD.printNumI(startnum,RIGHT,32);
Serial.println(averagetemp);
if (averagetemp>OPEN_TEMPERATURE)
{
      ifspray=1;      
}
else if (averagetemp<CLOSE_TEMPERATURE)
{
      ifspray=0;
}

//Serial.println(":");
//Serial.println(":");
//Serial.println(":");
if (ifspray==1)
{
      digitalWrite(RELAY_pump, HIGH);
      digitalWrite(RELAY_spray1, HIGH);
      delay(30000);
      digitalWrite(RELAY_spray1, LOW);
      digitalWrite(RELAY_spray2, HIGH);
      delay(30000);
      digitalWrite(RELAY_spray2, LOW);
      digitalWrite(RELAY_spray3, HIGH);
      delay(30000);
      digitalWrite(RELAY_spray3, LOW);
      startnum=startnum+1;
   }
   else
   {
      digitalWrite(RELAY_pump, LOW);
      delay(5000);
   }
}


wangnengjie 发表于 2014-9-12 11:08:10

补充,今天发现还是有读错的时候的。头疼。
还能不能好好玩耍了。。。

ardyPro 发表于 2014-9-18 08:44:58

baoz 发表于 2014-9-12 00:10 static/image/common/back.gif
搭车请教下LZ,DS18S20并联,读数的时候是什么样的反馈?有办法识别设备位置么?

DS18B20有全球唯一编号,根据它来判断

ardyPro 发表于 2014-9-18 08:46:59

wangnengjie 发表于 2014-9-11 14:22 static/image/common/back.gif
线的质量还是不错的,我也担心是太频繁操作的原因。只是想搞清楚这种稳定性的问题根源在哪里。
目前加了延 ...

按你之前的频率肯定不行,one wire的时序要求比较严格;可以试着把交流电电线跟信号线分开;降低上拉电阻阻值。

wangnengjie 发表于 2014-9-20 20:58:22

ardyPro 发表于 2014-9-18 08:46 static/image/common/back.gif
按你之前的频率肯定不行,one wire的时序要求比较严格;可以试着把交流电电线跟信号线分开;降低上拉电阻 ...

我加了延时5秒啊。还不行么?交流电基本上不通电的。打算加磁环试试的。手头还没找到合适的磁环。上拉电阻我原来用的100欧,后来怕小了,才换成1k的。100欧的时候一样有错误。
页: [1]
查看完整版本: 多DS18S20温度传感器运行时间长的问题。