我不确定最好的方法来做到这一点,并不是为我的工作,因为它将锁定程序时,它睡着了不确定如何使它正确工作。。。
我试着用树莓皮监控门的开和关,如果门开了超过x次,发送某种警报(比如和电子邮件),我有一个问题,当门在倒计时结束前关闭时,它不会停止倒计时,这会导致线程停止,也没有实现警报方面的东西,但如果代码是目前,它将触发警报,即使门关闭前倒计时。
目前,我正在使用一个按钮而不是一个门传感器进行测试,最终我也将记录门的打开和关闭,但现在我想知道是否有更好的方法来做这件事,我从这个post
我的代码如下
#!/usr/bin/python
import threading, subprocess, sys, time, syslog
import RPi.GPIO as GPIO
lim = 2 # seconds until warning
# thread for countdown (should be interruptable)
class CountdownTask:
global dooropen
global countdone
def __init__(self):
#print("thread in")
self._running = True
def start(self):
print("thread in")
self._running = True
def terminate(self):
print("thread killed")
self._running = False
def run(self, n):
while True:
global countdone
while self._running and dooropen == False and countdone:
pass
while self._running and dooropen == False and countdone == False:
pass
while self._running and dooropen and countdone:
pass
while self._running and dooropen and countdone == False:
print("start timer")
time.sleep(5)
if dooropen:
## action when timer isup
print("timer ended, send notify")
countdone = True
c = CountdownTask()
t = threading.Thread(target=c.run, args=(lim,))
t.daemon = True
REED = 23 # data pin of reed sensor (in)
# GPIO setup
GPIO.setwarnings(False)
GPIO.setmode(GPIO.BCM)
GPIO.setup(REED, GPIO.IN, pull_up_down=GPIO.PUD_DOWN)
dooropen = False # assuming door's closed when starting
countdone = True
def edge(channel):
global dooropen
global countdone
if GPIO.input(REED): # * no longer reached
if dooropen == False: # catch fridge compressor spike
print("Detect open")
countdone = False
dooropen = True
else:
print("Door closed")
dooropen = False
def main():
GPIO.add_event_detect(REED, GPIO.RISING, callback=edge, bouncetime=300)
t.start()
while True:
pass
#------------------------------------------------------------
if __name__ == "__main__": main()
更新:
看起来我需要使用线程。事件和等待有人能建议如何在我的代码上实现这一点吗?
最佳答案
我想我有个剧本
#!/usr/bin/python
import threading
import time
import logging
import RPi.GPIO as GPIO
import smtplib
from email.MIMEMultipart import MIMEMultipart
from email.MIMEText import MIMEText
logging.basicConfig(level=logging.DEBUG,format='(%(threadName)-9s) %(message)s',)
# GPIO setup
Input = 23 # data pin of Input sensor (in)
GPIO.setwarnings(False)
GPIO.setmode(GPIO.BCM)
GPIO.setup(Input, GPIO.IN, pull_up_down=GPIO.PUD_DOWN)
global button
button = False
global count
count = 1
#global t
t = 1
countdown = 5
def sendmail():
fromaddr = "email"
toaddr = "tomail"
msg = MIMEMultipart()
msg['From'] = fromaddr
msg['To'] = toaddr
msg['Subject'] = "DoorAlarm"
body = "This is a test mail generated with python on a RPi"
msg.attach(MIMEText(body, 'plain'))
server = smtplib.SMTP('smtp.gmail.com', 587)
server.starttls()
server.login("username", "password")
text = msg.as_string()
server.sendmail(fromaddr, toaddr, text)
server.quit()
def timeout(e, t):
global count
global button
while True:
while button:
while not e.isSet() and count <= countdown:
if count == 1: logging.debug('starting counter')
event_is_set = e.wait(t)
if event_is_set:
count = 1
logging.debug('Door closed before countdown')
else:
count += 1
if count == countdown:
logging.debug('countdown completed - notify')
#sendmail()
def edge(channel):
global button
global count
if button == False: # catch fridge compressor spike
button = True
e.clear()
print("log door open")
if count != 1:
count = 1
else:
button = False
e.set()
print("log door closed")
if __name__ == '__main__':
e = threading.Event()
t = threading.Thread(name='non-blocking',target=timeout,args=(e, t))
t.start()
logging.debug('Waiting before calling Event.set()')
GPIO.add_event_detect(Input, GPIO.RISING, callback=edge, bouncetime=300)
关于python - 如果GPIO输入更改,则倒计时线程RPi停止,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/38468557/