BYU logo Computer Science

To start this guide, download this zip file.

While loops

Imagine you start with an empty 5x3 Bit world…

empty 5x3 Bit world, with bit in the bottom left corner

and you want to fill the bottom row with green squares:

bottom row of the 5x3 world is green

You might imagine writing code like this:

from byubit import Bit


@Bit.empty_world(5, 3)
def go_green(bit):
    bit.paint('green')
    bit.move()
    bit.paint('green')
    bit.move()
    bit.paint('green')
    bit.move()
    bit.paint('green')
    bit.move()
    bit.paint('green')


if __name__ == '__main__':
    go_green(Bit.new_bit)

If you repeat bit.paint('green') and bit.move() enough times, you will eventually paint the bottom.

But what if your world was 100 squares wide?

100x3 blank world

The approach above would get really tedious!

While

This motivates why Python (and other programming languages) have a while loop:

def go_green(bit):
    while bit.can_move_front():
        bit.paint('green')
        bit.move()

This code says while the front of bit is clear, paint green and move forward one square.

We are introducing a new Bit function here:

bit.can_move_front()

This function returns True if the front of bit is clear. This will return True as long as the space in front of Bit is clear, otherwise it will return False.

Let’s look at the while loop more closely:

anatomy of a while loop

  • it must start with the while keyword
  • this is followed by a condition that evaluates to True or False
  • the first line ends with a colon :
  • the body of the while loop is indented

running a while loop

Every time Python runs the while loop it checks the condition. If it is true, then it runs the lines of code in the body of the loop. After it runs the body, it checks the condition again. It will keep checking the condition and run the body every time the condition is true. It leaves the loop if the condition is false.

While in action

To practice with while loops, download the zip file linked above. Find the go_green.py file, which looks like this:

from byubit import Bit


@Bit.empty_world(5, 3)
def go_green(bit):
    while bit.can_move_front():
        bit.paint('green')
        bit.move()


if __name__ == '__main__':
    go_green(Bit.new_bit)

Run it, and you should see this result:

result of running go_green.py, with green squares along the bottom, the last one left blank

  • click the First button
  • click the Next button repeatedly and watch the code execute

You should see Python check if the front is clear. This will be True the first time, so it will paint and then move. It will check the condition again … and repeat the loop, until it checks the condition and finds that the front is not clear, so it stops.

Boundary conditions

Every time you use a while loop, you should think about the boundary conditions:

  • Where is Bit when the loop starts?
  • Where will Bit be when the loop ends?
  • What squares will be colored?
  • Which squares will not be colored?
  • Which direction is Bit facing at the beginning? at the end?

In the code above, when Bit finishes the while loop, the last square is left unpainted.

Painting the last square

Remember, we want the whole bottom row to be green. How do we paint the last square green?

You need to add an extra bit.paint('green') outside the while loop:

def go_green(bit):
    while bit.can_move_front():
        bit.paint('green')
        bit.move()

    bit.paint('green')

Because this last bit.paint('green') is not indented, it is outside the while loop. When bit.can_move_front() is False, then Python will skip the loop and go to the next non-indented line. So it will paint one more square green and then finish the go_green() function:

this version of go green paints the entire bottom row green

Moving first and then painting

You could also change your go_green() function to move first and then paint green:

def go_green(bit):
    while bit.can_move_front():
        bit.move()
        bit.paint('green')

Here, the boundary conditions will be different! This will paint every square green except the first one. Step through the code using the First and Next buttons to see why.

this version of go green leaves the first square blank

You can paint the entire bottom row green by placing an extra bit.paint('green') that comes before the while loop:

def go_green(bit):
    bit.paint('green')
    while bit.can_move_front():
        bit.move()
        bit.paint('green')

It’s up to you to decide whether to paint first or move first. There is often more than one way to solve a problem.

Blocks

Bit worlds may sometimes include black squares:

a world with a black square in front of bit

Black squares stop bit from moving onto them. To see this in action, open the file called blocked.py, which is in the zip file you downloaded:

from byubit import Bit


@Bit.worlds('blocked')
def blocked(bit):
    bit.move()


if __name__ == '__main__':
    blocked(Bit.new_bit)

All bit does here is try to move one square. But if you run the code you will see this error:

bit is blocked from moving error

Conditions in Bit

Bit supports the following functions that check whether it is safe to move:

  • bit.can_move_front() — returns True if the front is clear, False otherwise
  • bit.can_move_right() — returns True if the right is clear, False otherwise
  • bit.can_move_left() — returns True if the left is clear, False otherwise

For a square to be clear, it must not be black or the edge of the world.

Looking at this world:

a world with a black square directly in front of bit

  • bit.can_move_front() is False
  • bit.can_move_right() is False
  • bit.can_move_left() is True

Look at the file called black_row.py, which is in the zip file you downloaded:

from byubit import Bit


@Bit.worlds('black-row')
def go(bit):
    while bit.can_move_right():
        bit.move()


if __name__ == '__main__':
    go(Bit.new_bit)

Bit starts in this world:

a 5 by 3 world with two black squares in the bottom row to the right

We want to move Bit until it is next to the first black square:

the same world with bit above the first black square

The code above uses while bit.can_move_right() to move until the right is no longer clear. Run the code above and use the First and Next buttons to see how it uses this condition.

Negating conditions

You can negate conditions using the not keyword. For example:

while not bit.can_move_right()
    bit.move()

will move Bit as long as the right side is not clear.

To see this in action, open the file called another-black-row.py, which is in the zip file you downloaded:

from byubit import Bit


@Bit.worlds('another-black-row')
def go(bit):
    while not bit.can_move_right():
        bit.move()


if __name__ == '__main__':
    go(Bit.new_bit)

Bit starts in this world:

a 5 by 3 world with three black squares in the bottom row to the left, bit directly above

We want to move Bit until it gets past all of the black squares:

the same world, with bit above the first blank square

The code above uses while not bit.can_move_right() to move until the right is clear. Run the code and use the First and Next buttons to see how it uses this condition.

Python will check bit.can_move_right() and this will be False. But then the not turns the False to True.

At the end of the loop, Python will check bit.can_move_right() and this will be True. But then the not turns the True to False, and so Bit leaves the while loop.

More conditions in Bit

Bit also provides a way for you to check which color of square Bit is currently on top of:

  • bit.is_on_blue() — returns True if Bit is on a blue square
  • bit.is_on_green() — returns True if Bit is on a green square
  • bit.is_on_red() — returns True if Bit is on a red square
  • bit.is_on_white() — returns True if Bit is on an empty square (no color)

To see this in action, open the file green_path.py, which is in the zip file you downloaded:

from byubit import Bit


@Bit.worlds('green-path')
def walk(bit):
    while bit.is_on_green():
        bit.move()


if __name__ == '__main__':
    walk(Bit.new_bit)

Bit starts in this world:

a 5 by 3 world with 3 green squares in the middle row to the left, bit on the first green square

We want to have Bit follow the green squares until it gets past the last one:

the same world, with bit on the blank square next to the green squares

The code above uses while bit.is_on_green() to move until the square it is on is not green. Run the code and use the First and Next buttons to see how it uses this condition.

You can find another example in blue_dot.py:

from byubit import Bit


@Bit.worlds('blue-dot')
def go_to_blue(bit):
    while not bit.is_on_blue():
        bit.move()


if __name__ == '__main__':
    go_to_blue(Bit.new_bit)

Bit starts in this world:

a 5 by 3 world with a blue square in the 2nd row and 4th column, bit starting in the second row on the left

We want to have Bit move until it gets to the blue square:

the same world, with bit on the blue square

The code above uses while not bit.is_on_blue() to move until the square it is on is blue. Run the code and use the First and Next buttons to see how it uses this condition.

Infinite loops

Open the file called infinite_loopy.py in the zip file you downloaded:

from byubit import Bit


@Bit.empty_world(3, 3)
def run(bit):
    while bit.can_move_front():
        bit.move()
        bit.paint('blue')
        bit.move()
        bit.paint('blue')
        bit.turn_left()


if __name__ == '__main__':
    run(Bit.new_bit)

This code works on an empty 3x3 world:

empty 3x3 world

In the code, while the front is clear, bit will:

  • move
  • paint a blue square
  • move
  • paint a blue square
  • turn left

You can draw this out and see that Bit will paint the bottom row blue and then turn left:

sketch of bit moving and painting blue

Since the front of bit is clear, it will do this again:

sketch of bit moving and painting blue again

and again and again and again …

sketch of bit moving and painting blue again

Thankfully, if you run the code, Bit will alert you if you have created an infinite loop:

infinite loop error