樹莓派 RPi.GPIO 使用說明
學習樹莓派的第一個模組就是 RPi.GPIO
,什麼是 GPIO ? GPIO(General Purpose I/O Ports)意思是通用輸入/輸出埠,也就是一些針腳可以通過它們輸出高電位或者低電位,可以供使用者由程式控制自由使用,與裝置進行通訊,達到控制裝置的目的。既然一個針腳可以用於輸入、輸出或其他特殊功能,那麼一定有暫存器用來選擇這些功能。對於輸入,一定可以透過讀取某個暫存器來確定針腳電位的高低;對於輸出,一定可以透過寫入某個暫存器來讓這個針腳輸出高電位或者低電位;對於其他特殊功能,則有另外的暫存器來控制它們,所以掌握了 GPIO 就相當於掌握了硬體的控制。
上圖有 40 根針腳,這就是樹莓派控制外部傳感器的接口,稱之為 GPIO(General Purpose I/O Port)。這裡可以看到有二種編號規則,分別是紅框內的 BOARD
編號規則,與紅框外的 BCM
編號規則,接下來我們就來試試 RPi.GPIO
的用法吧。
前置準備
教具準備
名稱 | 數量 | 規格 | 備註 |
---|---|---|---|
樹莓派 | 1 | Raspberry Pi 4 Model B (4G) | 就只是現在頂規是這個版本,用三代的也可以 |
麵包板 | 1 | 不限 | 測試時容易連結 |
杜邦線 | 1 | 不限 | 保證沒有斷芯就好 |
LED燈 | 2 | 不限 | 測試此設備能發光閃爍 |
電阻 | 2 | 10K | 為了不讓 LED 燒掉 |
檢查 wiringPi 版本與升級
由於樹莓派因版本不同,其各針腳的定義也會有些許的差異,那是否能在樹莓派裡就得知呢? 可以的,但你可能跟筆者一樣遇到 Pi 4B 版需要升級的情況,首先檢查一下版本。
$ gpio -v
回傳結果如下,是 2.50
版。
pi@raspberrypi:~ $ gpio -v
gpio version: 2.50
Copyright (c) 2012-2018 Gordon Henderson
This is free software with ABSOLUTELY NO WARRANTY.
For details type: gpio -warranty
Raspberry Pi Details:
Type: Unknown17, Revision: 02, Memory: 0MB, Maker: Sony
* Device tree is enabled.
*--> Raspberry Pi 4 Model B Rev 1.2
* This Raspberry Pi supports user-level GPIO access.
查看各針腳編號的命令如下:
$ gpio readall
回傳結果如下
pi@raspberrypi:~ $ gpio readall
Oops - unable to determine board type... model: 17
這個問題是 wiringPi
模組版本的問題,所以如果你也跟筆者一樣是 Pi 4B版且 wiringPi
版本是 2.50
版,那麼就跟著筆者按照官網指引升級即可,命令如下:
cd /tmp
wget https://project-downloads.drogon.net/wiringpi-latest.deb
sudo dpkg -i wiringpi-latest.deb
更新好後,再來檢查一次版本
$ gpio -v
這時候看到版本已更新到 2.52
了。
pi@raspberrypi:/tmp $ gpio -v
gpio version: 2.52
Copyright (c) 2012-2018 Gordon Henderson
This is free software with ABSOLUTELY NO WARRANTY.
For details type: gpio -warranty
Raspberry Pi Details:
Type: Pi 4B, Revision: 02, Memory: 4096MB, Maker: Sony
* Device tree is enabled.
*--> Raspberry Pi 4 Model B Rev 1.2
* This Raspberry Pi supports user-level GPIO access.
看看 Pi 4B 版的各腳位對應吧。
$ gpio readall
這時回傳的結果已可以看到 Pi 4B 版的各腳位編號了
pi@raspberrypi:/tmp $ gpio readall
+-----+-----+---------+------+---+---Pi 4B--+---+------+---------+-----+-----+
| BCM | wPi | Name | Mode | V | Physical | V | Mode | Name | wPi | BCM |
+-----+-----+---------+------+---+----++----+---+------+---------+-----+-----+
| | | 3.3v | | | 1 || 2 | | | 5v | | |
| 2 | 8 | SDA.1 | ALT0 | 1 | 3 || 4 | | | 5v | | |
| 3 | 9 | SCL.1 | ALT0 | 1 | 5 || 6 | | | 0v | | |
| 4 | 7 | GPIO. 7 | IN | 1 | 7 || 8 | 1 | ALT5 | TxD | 15 | 14 |
| | | 0v | | | 9 || 10 | 1 | ALT5 | RxD | 16 | 15 |
| 17 | 0 | GPIO. 0 | IN | 0 | 11 || 12 | 0 | IN | GPIO. 1 | 1 | 18 |
| 27 | 2 | GPIO. 2 | IN | 0 | 13 || 14 | | | 0v | | |
| 22 | 3 | GPIO. 3 | IN | 0 | 15 || 16 | 0 | IN | GPIO. 4 | 4 | 23 |
| | | 3.3v | | | 17 || 18 | 0 | IN | GPIO. 5 | 5 | 24 |
| 10 | 12 | MOSI | ALT0 | 0 | 19 || 20 | | | 0v | | |
| 9 | 13 | MISO | ALT0 | 0 | 21 || 22 | 0 | IN | GPIO. 6 | 6 | 25 |
| 11 | 14 | SCLK | ALT0 | 0 | 23 || 24 | 1 | OUT | CE0 | 10 | 8 |
| | | 0v | | | 25 || 26 | 1 | OUT | CE1 | 11 | 7 |
| 0 | 30 | SDA.0 | IN | 1 | 27 || 28 | 1 | IN | SCL.0 | 31 | 1 |
| 5 | 21 | GPIO.21 | IN | 1 | 29 || 30 | | | 0v | | |
| 6 | 22 | GPIO.22 | IN | 1 | 31 || 32 | 0 | IN | GPIO.26 | 26 | 12 |
| 13 | 23 | GPIO.23 | IN | 0 | 33 || 34 | | | 0v | | |
| 19 | 24 | GPIO.24 | IN | 0 | 35 || 36 | 0 | IN | GPIO.27 | 27 | 16 |
| 26 | 25 | GPIO.25 | IN | 0 | 37 || 38 | 0 | IN | GPIO.28 | 28 | 20 |
| | | 0v | | | 39 || 40 | 0 | IN | GPIO.29 | 29 | 21 |
+-----+-----+---------+------+---+----++----+---+------+---------+-----+-----+
| BCM | wPi | Name | Mode | V | Physical | V | Mode | Name | wPi | BCM |
+-----+-----+---------+------+---+---Pi 4B--+---+------+---------+-----+-----+
安裝 RPi.GPIO
不管用什麼方式,請在樹莓派的 Python 虛擬環境安裝 RPi.GPIO
模組,上述前置準備可以學習到怎麼連到此環境,安裝命令可以參考官方 RPI.GPIO 連結,命令如下 :
(dsalearning) $ pip install RPi.GPIO
安裝結果如下
(dsalearning) pi@raspberrypi:~/Documents/envs/dsalearning $ pip install RPi.GPIO
Looking in indexes: https://pypi.org/simple, https://www.piwheels.org/simple
Collecting RPi.GPIO
Using cached https://www.piwheels.org/simple/rpi-gpio/RPi.GPIO-0.7.0-cp37-cp37m-linux_armv7l.whl (69 kB)
Installing collected packages: RPi.GPIO
Successfully installed RPi.GPIO-0.7.0
看到 Successfully installed
就來檢查一下虛擬環境已匯入的模組是否有 RPi.GPIO
(dsalearning) $ pip list
檢查結果
(dsalearning) pi@raspberrypi:~/Documents/envs/dsalearning $ pip list
Package Version
---------- -------
pip 20.0.2
RPi.GPIO 0.7.0
setuptools 46.1.3
wheel 0.34.2
嗯,確定是已安裝進來了,可以放心的使用嘍~
RPi.GPIO方法與屬性
匯入模組
一起來了解 RPi.GPIO
1 這個模組的各個方法。匯入模組的指令如下:
import RPi.GPIO as GPIO
BOARD與BCM編號規則屬性
如上文針腳圖有紅框內的 BOARD
編號;與紅框外的 BCM
編號規則,此編號規則會因樹莓派版本而有些不同,所以使用時請務必留意。
打印出 BCM
編號規則屬性值
print(GPIO.BCM)
BCM
回傳結果如下
11
打印出 BOARD
編號規則屬性值
print(GPIO.BOARD)
BOARD 回傳結果如下
10
setmode()
上述知道編號規則屬性值各是多少了,但記數值不直覺,所以還是用屬性名稱吧!
設定 BCM 編號規則的程式如下
GPIO.setmode(GPIO.BCM)
設定 BOARD 編號規則的程式如下
GPIO.setmode(GPIO.BOARD)
getmode()
這個是取得被設置的編號規則屬性值
mode = GPIO.getmode()
回傳的值如下 :
- 未設定: None
- BCM: 11
- BOARD: 10
範例
查看未設定編號規則時是取得 None
,再設定指定的編號規則後,取得設定的屬性值。
import RPi.GPIO as GPIO
print(GPIO.getmode())
print(GPIO.BCM)
print(GPIO.BOARD)
GPIO.setmode(GPIO.BCM)
mode = GPIO.getmode()
print(mode)
回傳結果
None
11
10
11
setup()
在你要使用的針腳時,必須先設定這個針腳是作為輸出還是輸入。
語法
GPIO.setup({channel | chan_list}, {GPIO.OUT | GPIO.IN}, [initial= {GPIO.HIGH | GPIO.LOW}])
引數
channel
是指針腳的編號,依據設定的針腳編號規則的編號,例如
channel = 25 #BCM編號規則
chan_list
也可以一次設定多個針腳,例如
chan_list = [11,12]
GPIO.OUT
將指定的針腳作為輸出,如果想讓 LED 燈亮或驅動某個設備,就需要給它們電流和電壓,只要將針腳設定為輸出即可,範例如下
GPIO.setup(25, GPIO.OUT)
若去 print() 該屬性則會得到值為 0
,相當於是 boolean 值的 False,所以也可以用 False
填入,但為了讓程式的可讀性好,請還是使用屬性名稱的方式。
print(GPIO.OUT)
print(type(GPIO.OUT))
回傳結果如下
0
<class 'int'>
GPIO.IN
想要獲取針腳的狀態就將針腳設定為輸入,筆者認為這些傳感器輸出訊號,在樹莓派要接收就設為輸入。設定都跟上面的 GPIO.OUT
語法一樣。
initial
為針腳設定預設值,例如下方範例為設定輸出的針腳同時給予高電位。
GPIO.setup(channel, GPIO.OUT, initial=GPIO.HIGH)
相當於下面語法
GPIO.setup(channel, GPIO.OUT)
GPIO.output(channel,GPIO.HIGH)
output()
前面的範例已使用到 output()
,意思就是設定針對輸出的狀態是高電位還是低電位,這樣就可以讓 LED 燈亮或熄滅,或者驅動或停止驅動裝置。
GPIO.output(channel,GPIO.HIGH)
input()
當針腳已被設定高低電位時,可以用此函式取得該針腳的當下狀態為何,這樣就可以進一步有邏輯去判斷了。
if GPIO.input(channel):
print('HIGH')
else:
print('LOW')
cleanup()
好的程序猿就要懂得釋放資源,這樣才不會成為技術債或損壞樹莓派,釋放的命令如下:
GPIO.cleanup()
注意! GPIO.cleanup()只會清理執行腳本裡使用過的 GPIO 腳位,也會清除正在使用的編號規則。
setwarnings()
當檢測到某個針腳己經被設置為非預設值時,你會得到一個警告。如果要停用這個警告,就使用這個函式即可停止。
GPIO.setwarnings(False)
例如筆者的第一個範例 LED閃爍,第一次執行時並沒有警告出現,當再按一次執行時,就會出現如下的警告:
blink.py:7: RuntimeWarning: This channel is already in use, continuing anyway. Use GPIO.setwarnings(False) to disable warnings.
GPIO.setup(25,GPIO.OUT,initial=GPIO.LOW)
wait_for_edge()
待寫
event_detected()
待寫
範例
A. 使用 chan_list
設定針腳 24,25 (BCM編號) 為輸出,同時設定這二個針腳為輸出,本範例是將 2 個 LED 一開始都亮 3 秒後輪流亮 1 秒。
#實作 : 2個LED都亮3秒後輪流亮1秒
import RPi.GPIO as GPIO
import time
chan_list = [24,25]
GPIO.setmode(GPIO.BCM)
GPIO.setup(chan_list,GPIO.OUT)
GPIO.output(chan_list, GPIO.HIGH) #全部燈亮
time.sleep(3)
while True:
GPIO.output(chan_list, (GPIO.HIGH, GPIO.LOW)) #24 亮; 25 暗
time.sleep(1))
GPIO.output(chan_list, (GPIO.LOW, GPIO.HIGH)) #24 暗; 25 亮
time.sleep(1)
留言