树莓派作为一个完整的主机,除了能在上面跑系统外,也提供了40个引脚提供电路开发的功能。本来我在工作中常接触linux,但没啥机会接触硬件,正好手头上有一台raspberry 3b,本来是用来学linux相关的包的,闲的没事,网上买了几个传感器玩玩。很短时间就能上手从dht11读取数据了,虽然也看了好多文档,GPIO学起来还是挺简单的。
第一次写gpio程序, 我还是参考了别人的代码,主题代码基本一样,这里我多解释下我的代码吧。 重要的是这里有几个magic number,比如cnt 小于100和cnt > 12。
首先我们可以在网上找下DHT11的相关文档。你向传感器发送一个复位信号(大于18us的低电位,然后高电位),然后传感器会传送40-50us的低电位,紧接着40-50us的高电位。在往后就是以12-14us间隔的40个数据位了。
40个数据位编码方式如下。
byte4 byte3 byte2 byte1 byte0 01010101 00000000 10101010 00000000 01010101 -------- -------- -------- -------- -------- 整数 小数 整数 小数 校验位 ------------------ ------------------ 湿度 温度
这里有个校验位,是为了确保数据传输的准确性的。如果byte1+byte2+byte3+byte4 == byte0,数据传输就是正确的。不过DHT11小数位是不工作的,所以只需要考虑byte2+byte4就行了。
26-28us的高电平为0,116-118us的高电平为1。我代码里并没有以时间间隔作为判断,而是用了cnt 这个while循环计数器,因为我尝试过用时间。测试得到一次while空循环带计数器+1,基本上需要0.8us左右,但如果while循环里加上接口数据读取,一次循环基本上就要6us左右了,果然外部IO的速度比较慢。
这里我经过多次测试,发现计数器cnt>=12作为1的判断准确率较高。这就是我magicnumber 12的由来。 还有另外一个magicnumber 100,这个数字主要是用来应对复位信号没发送成功,程序陷入死循环的情况。我用的是树莓派3B,不同版本的树莓派性能不一样,可能magicnumber也不一样。
我用的是BCM模式的4引脚,也就是板子上的7号引脚作数据传输,另外两根引脚接地和3.3v电压即可。
下面就是代码了
#!/usr/bin/python import RPi.GPIO as GPIO import time pin = 4 data = [] GPIO.setmode(GPIO.BCM) time.sleep(2) GPIO.setup(pin, GPIO.OUT) GPIO.output(pin, GPIO.HIGH) GPIO.output(pin, GPIO.LOW) time.sleep(0.04) GPIO.output(pin, GPIO.HIGH) GPIO.setup(pin, GPIO.IN) while GPIO.input(pin) == GPIO.LOW: continue while GPIO.input(pin) == GPIO.HIGH: continue i = 0 while i < 40: cnt = 0 while GPIO.input(pin) == GPIO.LOW: continue while GPIO.input(pin) == GPIO.HIGH: cnt += 1 if cnt > 100: break if cnt < 12: data.append(0) else: data.append(1) i += 1 humidity_bit = data[0:8] humidity_point_bit = data[8:16] temperature_bit = data[16:24] temperature_point_bit = data[24:32] check_bit = data[32:40] humidity = 0 humidity_point = 0 temperature = 0 temperature_point = 0 check = 0 for i in range(8): humidity += humidity_bit[i] * 2 ** (7-i) humidity_point += humidity_point_bit[i] * 2 ** (7-i) temperature += temperature_bit[i] * 2 ** (7-i) temperature_point += temperature_point_bit[i] * 2 ** (7-i) check += check_bit[i] * 2 ** (7-i) tmp = humidity + humidity_point + temperature + temperature_point if check == tmp: print "temperature :", temperature, "*C, humidity:", humidity, "%" else: print "wrong" GPIO.cleanup()