Code Toolkit: Python, Spring 2024

Week 6 — Wednesday, February 28 — Class notes

  1. Review: loops
  2. New concept: working with many things
  3. Background and examples
  4. Lists

I. Review: loops

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: the initial variable assignment of 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 took a look at Leen's homework (Thanks, Leen, for sharing!) to see a few things:

  1. Using two independent loops — notice how this creates repetition linearly.
  2. Using two loops with one nested inside the other — notice how this creates repetition in two dimensions.
  3. Using nested loops to create a Sol Lewitt experiment. Nice work!

Working with the first two parts above, we talked about how you can use modulo to create alternating colors for elements drawn within loops. Here's how to do that with a single, linear loop, modifying part 1 above:

i = 50
while i <= 550:
    if i % 100 == 0:
        fill(100,210,0)
    else:
        fill(210,100,0)
    rect(i,300, 20,20)
    i = i + 50

And here is how to do that with a nested loop, modifying part 2 above:

i = 50
while i <= 500:
    
    j = 50
    while j <= 500:
        if (i+j) % 100 == 0:
            fill(100,210,0)
        else:
            fill(210,100,0)

        rect(j,i, 40,40)
        j = j + 50

    i = i + 50

lerpColor(). We didn't talk about this in class but loops allow you to use several different techniques for creating interesting color gradients. Many of you in your homework used using looping variables with the fill() command to acheive this. Another technique is to use looping variables with lerpColor(), which allows you to create a gradient that fades smoothly between two previously defined colors. Again, we didn't discuss this in class, but here's an example of how you might use lerpColor() in a loop to create a gradient:

start_color = color(252,98,255,150)
end_color = color(250,174,103,150)
i = 0
while i < 25:
    c = lerpColor(start_color,end_color,i/25.0)
    fill(c)
    ellipse(350,100+i*20,200,30)
    i = i + 1
And here is how you might try a similar technique in a nested loop:
start_color = color(252,98,255,150)
end_color = color(250,174,103,150)
i = 0
while i < 50:
    j = 0
    while j < 50:
        c = lerpColor(start_color,end_color,(i+j)/100.0)
        fill(c)
        ellipse(100+j*10,100+i*10,12,12)
        j = j + 1
    i = i + 1
If you'd like to experiment with this feel free to reach out with any questions.

(jump back up to table of contents)

II. Working with may things

Today we will learn about working with many things.

Last week (week 5) we learned how to draw many items using loops, but you couldn't move each item around individually. Thus far, if you wanted to move many things, you would have had to hard code new variables for each one. Today we will learn how to draw many things and move many things, using a thing called lists.

In Python, a list is like a collection of variables. In a 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 topic is an introduction to one of the most interesting and important 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 "built-in" to 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 a certain sense, lists are the most fundamental data structure: we could think about ways to implement all these other more complicated data structures using just lists, so maybe we can think about lists 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)

III. Background and examples

Gene Kogan

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, using a well-known algorithm developed by Craig Reynolds known as "Boids".

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)

IV. Lists