Last week we
covered loops, and in
particular, while
loops.
Loops allow us to create repetition, which in coding is often called iteration.
Prior to last week we'd already seen one kind of repetition:
the def draw()
block, which runs
all the code in that block and then repeats many times per
second. (The number of repetitions per second is defined by
the frame rate, and indicated by the special
variable frameRate
.) But this repetition
with def draw()
unfolds in time, and so is not
typically referred to as iteration.
The repetition that we call iteration is how you would go from drawing one thing on the screen to drawing several, or hundreds, or millions, or a dynamic number of things that you don't know in advance.
The syntax for while
loops looks like this:
i = 0 while i < 10: println("i = " + str(i)) rect(i,i,5,5) i = i + 1
Which is comprised of: a variable declaration (what I call the looping variable), a boolean expression, and a variable increment.
If the boolean expression in
the while
statement returns True
, all
the code in the while
block will
run, and then Python will check the boolean
expression again. If the expression is
still True
, the code will run again. And the code
keeps running as long as the expression
returns True
, or we might say while it
is True
.
Remember: Don't forget your variable incremeent or you might end up with an infinite loop, which repeats forever without ever stopping.
I also talked briefly about how while
loops are
functionally equivalent to for
loops, which offer
you an option of a different syntax that effectively does the
same thing. for
loops look like this:
for i in range(10): println("i = " + str(i)) rect(i,i,5,5)There are times when you may need to use one type of loop over the other, but in general these are equivalent and you can use the one that is more clear to you.
Homework review. We looked at this code (Thanks, Lexi, for sharing!) and debugged it together: week06_hw_inclass.pyde.txt
That code demonstrates some experiments in creating color
gradients within a nested loop. The code shows two main
techniques for how to do this: using looping variables with
the fill()
command to generate a gradient using an
RGB color specification, and using looping variables
with lerpColor()
to create a gradient that fades
between two previously defined colors. The difference techniques
in that file (three including a 2b option) are mutually
exclusive. In other words, uncomment one to see what it does and
experiment, then comment that one out and uncomment another to
experiment with it.
Today we will learn about working with many things. So far if you wanted to move many things, you would hard code new variables for each one. In week 5 we learned how to draw many items using loops, but you couldn't move each item around individually. Today we will learn how to do that using a thing called lists.
In Python, a list is like a collection of variables.
In a certain way, this topic is like a mashup of variables and loops:
With loops, instead of drawing many things line-by-line, you can create a loop which draws many things at once, even a dynamic number of things.
With lists, instead of declaring each variable line-by-line, you can create one list that declares many variables at once, and can even allow you to work with a dynamic number of variables.
This introduces us to one of the most interesting aspects of computer science: data structures. The term data structure refers to a way of organizing many variables together for efficient and convenient use. The type of variable organization that you'll need to use in a given situation (i.e., the type of data structure) depends on the problem that you are trying to solve. In addition to lists, Python provides many other data structures, such as: sets, tuples, and dictionaries, among others. And beyond these data structures that come with Python, in computer science there are many other more complex kinds of data structures that go by names like: stacks, queues, linked lists, and trees. But in one sense, lists are the most fundamental data structure, and in a certain way it makes sense to think of them as underlying all these others.
Data structures allow you to start to think about modeling: how to organize your variables and other code in a way that matches the thing that you're trying to do. A good data structure could make your program much easier to implement and to read, while an inappropriately chosen data structure could make the thing that you're trying to do very hard and potentially less efficient in terms of computing resources.
(jump back up to table of contents)Since lists are used to manage many things, they are often used to implement things like swarms, or herds of objects that behave as if they are acting independently or on their own.
Daniel Shiffman, a professor at NYU in the Interactive Telecommunications Program (ITP) graduate program, uses lists and other data structures to implement swarms that are used to simulate natural phenomenon like animals, plants, clouds, flowing liquids, and others.
For example, here is a video that demonstrates a simulation of "flocking", like birds or fish moving together. (It's a very cute video.) Shiffman's book Nature of Code offers detailed lessons in how to achieve affects like this.
Another, more aesthetically developed, example is this flocking demo by Gene Kogan.
And another great example is the project We Feel Fine by Jonathan Harris and Sep Kamvar. This project was made in 2006 using Processing and unfortuntely can be a bit difficult to run that now. (You will probably need to install Java for your browswer and OS version.) But there is documentation online, like this video that demonstrates how the project functioned.
(jump back up to table of contents)
When we first looked at variables, we saw how you could use
them to add "variance" or "change" into your
composition. You replace a hard-coded
number with a placeholder like x
, and then you
could change the value of that placeholder, modifying your
sketch. And each variable stores one value.
Back during week 2, when variables
were introduced, we talked about how many variables you
would need to implement the game Pong. (Then
during week 4 we saw how to
implement this using if
statements and keyboard
interaction.)
Do you remember how many variables we decided we would need to implement Pong? Remember that the ball moves in two directions, horizontal and vertical, there are two paddles that each move vertically, and there are two scores:
ballX
,ballY
,leftPaddleY
,rightPaddley
,scoreLeftPlayer
, and scoreRightPlayer
.x = x +
1
and instead sometimes might be x = x -
1
, to move to the left. And because of this, we need
a variable for the ball direction in horizontal and vertical
dimensions. So:
ballXDirection
, and
ballYDirection
What if you wanted 2 ping-pong balls? You could add another four variables (for x, y, x direction, y direction). But what if you wanted 3, or hundreds? What if you wanted a dynamic number? How could you haver a dynamic number of variables? For this, instead of regular variables for x and y position and direction, you would use lists.
Have a look at this example if you'd like, but don't be intimidated. There's a lot going in there, and we'll work through it today bit-by-bit.
def setup(): size(500, 500) fill(150, 150, 250) rectMode(CENTER) def draw(): background(255) rect(100, 250, 100, 100) rect(200, 250, 100, 100) rect(300, 250, 100, 100) rect(400, 250, 100, 100)
Now what if we want each one of these squares to move up and down randomly? Well to start, let's just add a variable for each one:
a = 200 b = 200 c = 200 d = 200 def setup(): size(500, 500) stroke(50, 50, 250) fill(150, 150, 250) rectMode(CENTER) def draw(): background(255) rect(100, a, 100, 100) rect(200, b, 100, 100) rect(300, c, 100, 100) rect(400, d, 100, 100)Nothing new yet, I've just swapped out a hard-coded number for a variable placeholder. Note that I don't need to add
global
inside the draw()
block
yet because while I am using these global
variables, I am not assigning or modifying
them.
Now to make each square move, we could change the value
of each variable in draw()
. Let's change it by
a random amount, so they just kind of shake there.
a = 200 b = 200 c = 200 d = 200 def setup(): size(500, 500) stroke(50, 50, 250) fill(150, 150, 250) rectMode(CENTER) def draw(): global a, b, c, d background(255) rect(100, a, 100, 100) rect(200, b, 100, 100) rect(300, c, 100, 100) rect(400, d, 100, 100) a = a + random(-5,5) b = b + random(-5,5) c = c + random(-5,5) d = d + random(-5,5)Note that now I am modifying the variables inside the
draw()
block, so now I do have to
add global
because otherwise Python would think that
my assignment statements were creating new local
variables. Hopefully this global
is getting clearer
for you, but I agree that it can be confusing.
Looking at that code, notice that I'm doing the same thing
several times: drawing a rect()
. And the
paramters to each one follow a simple and direct
pattern. Hopefully that makes you want to replace those four
lines with something else. What is it? Think back
to week 5 ... What if we wanted to
have 8 or 100 or 1000 squares?
So hopefully you realized that we could replace those
four rect()
commands with
one while
loop that draws all four:
a = 200 b = 200 c = 200 d = 200 def setup(): size(500, 500) stroke(50, 50, 250) fill(150, 150, 250) rectMode(CENTER) def draw(): global a, b, c, d background(255) i = 100 while i <= 400: rect( i, a, 100, 100) i = i + 100 a = a + random(-5, 5) b = b + random(-5, 5) c = c + random(-5, 5) d = d + random(-5, 5)If you step through that loop, you'll see that it is replicating the four
rect()
statements from the
previous snippet.
BUT WAIT — what about the
variables b
, c
,
and d
?! As you can see from running that, the
vertical position of each square is now being controlled
by a
. Changing the value of a
changes the position of each square in the same way. But we
can't use a
, b
, c
,
and d
independently because we are in a loop.
This is precisely why we need lists. To keep track of many things in a situation like a loop.
And we might even have a dynamic number of
things. Think for example what would happen if I
used mouseX
in my while
conditional.
So what we're about to do is: replace those four variables with one variable, a list, and that one list variable will hold all four values within it and will let us reference them inside a loop.
x = []
These square brackets create a new
kind of variable, which is a list. It
does not have any values yet, but we've just told
Python that this one variable can hold many values. I
suggest reading this code like: "Create a variable called
x and set it to an empty list".
Now to actually add values to this new list variable, we
use a command called append()
. Like this:
x.append(5) x.append("Hello") x.append(True)As you see, a list can contain any of the other values that we've been working with so far: numbers, text strings, and Boolean values. You can view the contents of a list with
print()
:
print(x) # This would print to the console: # [5, 'Hello', True]
But the really new, weird, and exciting thing is how we actually reference, or in other words, how we use those values.
Lists are ordered. The values they hold are stored in order. And you reference those values by number, using a new syntax, square brackets:
print( x[0] ) # This would print to the console # 5This is called the list index. The word index (like the index in a book or your index finger) has etymology related to pointing. So you can think of the
0
here as pointing to a
specific value, the first value. (In other parts of
computer science, this is actually called
a pointer,
but pointers are a slightly different
and more complicated topic common in C and C++
programming.)
You can use an indexed list anywhere that you would use a regular variable, so any of the following would be valid:
rect( x[0], 5, 100, 100) fill( 155, 155, x[0] ) if x[5] < 10: # do something(Obviously this snippet wouldn't make any sense altogether.)
You also use the assignment
operator =
to set values in a list
like this:
x[0] = 5But you cannot use the assignment operator to add new values to the list, or in other words to make the list longer. For example, this will get an error:
y = [] # Make a new empty list y[0] = 5 # Try to set a value into the list # Displays an error in the console saying: # IndexError: list assignment index out of range # But this would work: y.append(5) # And then you could change the value later like this: y[0] = 6
The exciting thing is that now we have a list of values, and they are referenced or indexed using a number. That allows us to do things like reference those values in a loop or have a dynamic number of values.
Important note. You may have noticed something
a little weird: lists are
always indexed starting with 0. So the
first item of this list would be x[0]
, and
if a list had ten items, the last would be
indexed as x[9]
.
You can ask Python for the length of a list like
this: len(x)
. This is a very important and
extremely common thing to do.
x
ranges from 0
to len(x)-1
Let's put this to use in our example, but start by first by
going back to the version without a while
loop:
y = [] y.append(200) y.append(200) y.append(200) y.append(200) def setup(): size(500, 500) stroke(50, 50, 250) fill(150, 150, 250) rectMode(CENTER) def draw(): background(255) rect(100, y[0], 100, 100) rect(200, y[1], 100, 100) rect(300, y[2], 100, 100) rect(400, y[3], 100, 100) y[0] = y[0] + random(-5,5) y[1] = y[1] + random(-5,5) y[2] = y[2] + random(-5,5) y[3] = y[3] + random(-5,5)We're using a list! Notice that instead of creating four independent variables,
a
, b
, c
,
and d
, I am now creating
one list called y
, and
appending the value 200
to it four times. And
then I reference those values using the square
bracket index notation, as
items 0
, 1
, 2
,
and 3
. So instead of saying a
, I'm
saying y[0]
.
This doesn't have any apparent advantage yet, but let's keep going ...
Remember that my goal was to use a loop to draw and move my squares, so that I could have a dynamic number. Here's where it gets exciting and a little tricky.
Since we use numbers to index lists, we can also use a variable for the list index. In other words, we can use a variable to specify which item in the list we are referring to. For example:
n = [] n.append(350) i = 0 print( n[i] ) # Prints to the console: # 350Pause and make sure you can wrap your head around that. What is the value of
i
?
Step through this example:
clouds = [] clouds[0] = 700 clouds[1] = 800 clouds[2] = 900 i = 0 print( clouds[i] ) # What would this print? 700 (Highlight to see.) i = i + 1 print( clouds[i] ) # What would this print? 800
Now that we can use variables to index our list, we're almost there. Let's go back to our loop, but we have to change one thing first.
Remember how last week we talked about different ways of
specifying our looping varible in a while
loop?
Compare the below two examples to see how they are equivalent:
i = 100 while i <= 400: rect(i, a, 100, 100) i = i + 100
i = 0 while i <= 3: rect(100 + i*100, a, 100, 100) i = i + 1
We can finally put everything together:
y = [] y.append(200) y.append(200) y.append(200) y.append(200) def setup(): size(500, 500) stroke(50, 50, 250) fill(150, 150, 250) rectMode(CENTER) def draw(): background(255) i = 0 while i <= 3: rect(100 + i*100, y[i], 100, 100) i = i + 1 y[0] = y[0] + random(-5,5) y[1] = y[1] + random(-5,5) y[2] = y[2] + random(-5,5) y[3] = y[3] + random(-5,5)We can also move the code that changes the y values into the same loop:
y = []
y.append(200)
y.append(200)
y.append(200)
y.append(200)
def setup():
size(500, 500)
stroke(50, 50, 250)
fill(150, 150, 250)
rectMode(CENTER)
def draw():
background(255)
i = 0
while i <= 3:
rect(100 + i*100, y[i], 100, 100)
y[i] = y[i] + random(-5,5)
i = i + 1
And we can also make a loop that initializes the list
values. Note that I've moved the first 5 lines of the code
snippet above into the setup()
block and replaced
the last 4 lines with a loop:
def setup(): size(500, 500) stroke(50, 50, 250) fill(150, 150, 250) rectMode(CENTER) global y y = [] i = 0 while i <= 3: y.append(200) i = i + 1 def draw(): background(255) i = 0 while i <= 3: rect(100 + i*100, y[i], 100, 100) y[i] = y[i] + random(-5,5) i = i + 1Note that since I am creating
y
inside
the setup()
block (by saying y =
[]
) I need to declare it global
so that
it can be accessed inside the draw()
block.
Now we have a loop that uses repetition to create many things and and one list that holds many values, one for each of those things.
This means that now we could relatively easily modify this
so that instead of 4 squares, we had 8, 100, or 1000. Change
the 3
to 4
to see how easy it is
to add one more:
def setup(): size(500, 500) stroke(50, 50, 250) fill(150, 150, 250) rectMode(CENTER) global y y = [] i = 0 while i <= 4: y.append(200) i = i + 1 def draw(): background(255) i = 0 while i <= 4: rect(100 + i*100, y[i], 100, 100) y[i] = y[i] + random(-5,5) i = i + 1Or change it to
10
(for this, I'll make
each one less wide):
def setup(): size(500, 500) stroke(50, 50, 250) fill(150, 150, 250) rectMode(CENTER) global y y = [] i = 0 while i <= 10: y.append(200) i = i + 1 def draw(): background(255) i = 0 while i <= 10: rect(100 + i*25, y[i], 25, 100) y[i] = y[i] + random(-5,5) i = i + 1
What if we only wanted to move the even-numbered rectangles? We
could add an if
statement inside that last loop:
i = 0
while i <= 3:
rect(100+i*100, y[i], 100, 100)
if i == 0 or i == 2:
y[i] = y[i] + random(-5,5)
i = i + 1
But then if we wanted to change our list size, we'd
have to keep adding additional checks into
that if
statement.
So to do this more concisely, we could use a thing called
the modulo which is like division, but only
returns the remainder.
Here are some examples that help demonstrate how modulo works:
# Regular division: print( 10 / 2 ) # Prints 5 # The remainder when dividing 10 by 2: print( 10 % 2 ) # Prints 0 # Python ignores the decimal when you're using whole numbers: print( 10 / 3 ) # Prints 3 # Using the decimal tells Python not to ignore it: print( 10.0 / 3 ) # Prints 3.3333 # The remainder when dividing 10 by 3: print( 10 % 3 ) # Prints 1 # 10 goes in to 11 once, with remainder 1: print( 11 % 10 ) # Prints 1 # So this expression would print True whenever x was an even number: print( x % 2 == 0 )And here's how you could use that in the above loop:
i = 0
while i <= 3:
rect(100+i*100, y[i], 100, 100)
if i % 2 == 0: # if dividing by 2 gives remainder 0, it's even
y[i] = y[i] + random(-5,5)
i = i + 1
while
loop can also
be interactive based on mouse input:
i = 0
while i <= 3:
rect(100+i*100, y[i], 100, 100)
if mouseX > i*100-50 and mouseX < i*100+50:
y[i] = y[i] + random(-5,5)
i = i + 1
(If you're working through the math on that remember that
I'm using rectMode(CENTER)
in these
examples.)
To conclude, what should we do if we want these squares to move around in space, like in the flocking examples? In other words, to move horizontally as well as vertically?
We would need an x value for each square. So? Add a new list!
x = [] # in setupAnd go from there: set initial values, use them with the
rect()
command, and change the values in
some way.
If you wanted each square to have other properties (like size and color), simply add more lists in the same way. Here is an example in which each square has its own x and y position, as well as size and color: list_example.pyde
If you wanted them to move in a way that was more sophisticated than just random, what would you do? Well each square would need its own x and y direction. More lists!
Here's a basic version of the game Breakout that puts it all together: breakout.pyde