adventures with microcomputers prototyping with raspberry pi

10. GPIO: Blink



Install the gpiozero library: sudo apt-get install python3-gpiozero


  • 1x Resistor (270 or 330 Ohms or similar)
  • 1x LED

hookup pattern

Note: Illustrations sourced from here.

Confirm that you have a functional LED by wiring it up inline with a resistor between 3V3 and GND. See below for an illustration of this hookup pattern:

After confirming that your LED works, move the wire that was previously attached at 3V3 to GPIO pin 17 (labeled as G17 on the PiWedge). See below for an illustration of this hookup pattern:

on the way to blink

Rather than scp-ing files between our Macbook and Pi it's more convenient to clone my rpi github repository to your Pi: git clone

Once the rpi repo is on your Pi cd into the scripts folder inside of the 01_blink folder: cd rpi/05_GPIO/01_blink/scripts/


One can print the contents of a folder to the terminal with the Unix command ls. Execute ls in the scripts directory now.

There are other parameters that one can use with ls to alter its behavior. One command that you should commit to memory is ls -la. Execute ls -la in the scripts directory now.

ls -la shows all (or -a) of the contents of a folder in long (or -l) format, which includes lots of useful information: the exact size of the file, who owns the file, who has the right to look at it, and when it was last modified.

Right now we just need to focus on the permissions component of the ls -la output, which generally looks like the following: -rw-r--r--. For our current purposes all we need to understand is that such a file is not executable, though it needs to be in order to be run via python.

Make the scripts in this folder executable by editing (to suit your needs) and executing the following line per file: sudo chmod +x

Confirm that the files are executable by running ls -la in the scripts directory once more. All we care about right now is that the permissions line ends with an x like so: -rwxr-xr-x

try, except, and handling errors

errors are one's friend when programming, as they give a programmer clues towards what is problematic about his or her code. Frequently we discuss Syntactical Errors, or those errors caused by typos or misspellings. Nonetheless, it's also possible to encounter errors during execution of a script. These errors are referred to as Exceptions and may or may not be ignorable (errors referred to as fatal are those errors which are not ignorable).

It is considered best practice to anticipate the errors one's code may encounter during execution and design a way for the program to handle those errors. The try statement specifies exception handlers for a group of statements. In other words, one tries to execute some code with a try statement and then provides various procedures to handle errors as listed in the except statement.

The simplest example of handling a non-fatal error can be seen below:

    while True:
        print("do stuff!")
except KeyboardInterrupt:

In the above example we have the basic structure of a try except block that handles a KeyboardInterrupt (i.e. ). In other words, one can put a line of code to happen forever in a infinite while loop (while True:) and simply end the loop by hitting .

Another example (from the code) can seen below:

from gpiozero import LED
from time import sleep

led = LED(17)

    while True:
except KeyboardInterrupt:
        led.close() # free Pin 17

In the above example an LED blinks on and off forever, however as soon as a KeyboardInterrupt occurs the while loop is terminated and the program cleans up the pin allocated to the LED (led.close()).

At a minimum I add KeyboardInterrupt as an exception in Python code, particularly when prototyping.

simple blink

  1. Run ipython3
  2. After reveling in the wonder of a Python3 version of Arduino's blink example, terminate the while loop with .
  3. Try changing the duration of either or both sleep lines to increase/decrease the amount of time the LED is on/off and run the script again.

blink for a random duration is a slightly more complex version of A few notes:

  • we import the random module to gain access to various random number generators. Currently we use random.uniform() to generate a number between 0.1 and 3 (storing that number in interval for use in both sleep statements) but there are lots of other generators provided by the random module. One can leverage iPython3's .<TAB> functionality to list all methods associated with random. Even better, after locating a generator that sounds promising, bring up the help docs by adding a ? to the end of the command and hitting .
  • we concatenate (combine) a string statement on line 12 with the following: print( "interval is " + str(interval)). This give one a way to see the labeled numerical output of random.uniform() in the terminal. One has to string cast the value stored in interval from floatto string prior to concatenation.
  • the lazy way to insert newlines between blocks of print() statements is to included an empty print(), which occurs here on line 17. There is probably a better way to do this.
  • led.close() is used here as part of the exception block in our try statement, allowing one to simultaneously terminate the while loop and cleanup pin allocations.