はじめに
ラズパイで環境情報を取得するために、DHT11温湿度センサを使っています。
取得した最新の温湿度の情報を「2021/07/22 23:00:01 20 82」というような形でファイルに保存しているのでその方法をまとめておきます。
センサーのつなげ方
DHT11センサにはピンが4本あるものと3本のものがありそうです。
私が持っているのは3本のもので、左からVCC,GRD,DATAとなっていました。
VCCは4番ポート(5V)、GRDは6番ポート(GRD)、DATAは8番ポート(GPIO14)に接続しました。
温湿度情報の処理
温湿度をファイルに出力する処理は以下です。OSOYOOのサンプルコードをベースにしているので細かいところはわかりませんが、処理を実行すると「/var/local/NowTemp.txt」に最新の日時と温湿度を出力します。
import RPi.GPIO as GPIO
import time
import os
import datetime
#GPIOの番号
DHTPIN = 14
GPIO.setmode(GPIO.BCM)
MAX_UNCHANGE_COUNT = 100
STATE_INIT_PULL_DOWN = 1
STATE_INIT_PULL_UP = 2
STATE_DATA_FIRST_PULL_DOWN = 3
STATE_DATA_PULL_UP = 4
STATE_DATA_PULL_DOWN = 5
def read_dht11_dat():
GPIO.setup(DHTPIN, GPIO.OUT)
GPIO.output(DHTPIN, GPIO.HIGH)
time.sleep(0.05)
GPIO.output(DHTPIN, GPIO.LOW)
time.sleep(0.02)
GPIO.setup(DHTPIN, GPIO.IN, GPIO.PUD_UP)
unchanged_count = 0
last = -1
data = []
#センサーからデータを取得
while True:
current = GPIO.input(DHTPIN)
data.append(current)
if last != current:
unchanged_count = 0
last = current
else:
unchanged_count += 1
if unchanged_count > MAX_UNCHANGE_COUNT:
break
state = STATE_INIT_PULL_DOWN
lengths = []
current_length = 0
#温度と湿度が混ざった40個のリストを作成
for current in data:
current_length += 1
if state == STATE_INIT_PULL_DOWN:
if current == GPIO.LOW:
state = STATE_INIT_PULL_UP
else:
continue
if state == STATE_INIT_PULL_UP:
if current == GPIO.HIGH:
state = STATE_DATA_FIRST_PULL_DOWN
else:
continue
if state == STATE_DATA_FIRST_PULL_DOWN:
if current == GPIO.LOW:
state = STATE_DATA_PULL_UP
else:
continue
if state == STATE_DATA_PULL_UP:
if current == GPIO.HIGH:
current_length = 0
state = STATE_DATA_PULL_DOWN
else:
continue
if state == STATE_DATA_PULL_DOWN:
if current == GPIO.LOW:
lengths.append(current_length)
state = STATE_DATA_PULL_UP
else:
continue
if len(lengths) != 40:
return False
halfway = (max(lengths) + min(lengths)) / 2
bits = []
the_bytes = []
byte = 0
#中間より多いか少ないかで温度と湿度を判別している
for length in lengths:
#中間より小さければ温度
bit = 0
if length > halfway:
#中間より大きければ湿度
bit = 1
bits.append(bit)
#[湿度,XX,温度,XX,チェックサム]のリストの作成
for i in range(0, len(bits)):
byte = byte << 1
if (bits[i]):
byte = byte | 1
else:
byte = byte | 0
if ((i + 1) % 8 == 0):
the_bytes.append(byte)
byte = 0
#チェックサム比較
checksum = (the_bytes[0] + the_bytes[1] + the_bytes[2] + the_bytes[3]) & 0xFF
if the_bytes[4] != checksum:
return False
return the_bytes[0], the_bytes[2]
def main():
#温湿度情報を取得
while True:
result = read_dht11_dat()
#取得できるまで繰り返し
if result != False:
humidity, temperature = result
break
#日付情報を取得
date = datetime.datetime.now()
#温湿度情報をファイル出力
output = date.strftime('%Y/%m/%d %H:%M:%S ' + str(temperature) + " " + str(humidity))
com = "echo \"" + output + "\" > /var/local/NowTemp.txt"
os.system(com)
def destroy():
GPIO.cleanup()
if __name__ == '__main__':
try:
main()
except KeyboardInterrupt:
destroy()
10分に1回温度情報を更新
常に最新の温度にしておきたいので、10分周期でコマンド実行するように/etc/crontabに記載して、「systemctl resatar cron」でcronを再起動します。
すると10分周期で最新情報に更新してくれます。
# Example of job definition:
# .---------------- minute (0 - 59)
# | .------------- hour (0 - 23)
# | | .---------- day of month (1 - 31)
# | | | .------- month (1 - 12) OR jan,feb,mar,apr ...
# | | | | .---- day of week (0 - 6) (Sunday=0 or 7) OR sun,mon,tue,wed,thu,fri,sat
# | | | | |
# * * * * * user-name command to be executed
*/10 * * * * root python3 /home/pi/leo_test/temp_check2.py