To start this guide, download this zip file.
While loops
Imagine you start with an empty 5x3 Bit world…
and you want to fill the bottom row with green squares:
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?
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:
- 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
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:
- 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:
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.
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:
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:
Conditions in Bit
Bit supports the following functions that check whether it is safe to move:
bit.can_move_front()
— returnsTrue
if the front is clear,False
otherwisebit.can_move_right()
— returnsTrue
if the right is clear,False
otherwisebit.can_move_left()
— returnsTrue
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:
bit.can_move_front()
isFalse
bit.can_move_right()
isFalse
bit.can_move_left()
isTrue
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:
We want to move Bit until it is next to 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:
We want to move Bit until it gets past all of the black squares:
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()
— returnsTrue
if Bit is on a blue squarebit.is_on_green()
— returnsTrue
if Bit is on a green squarebit.is_on_red()
— returnsTrue
if Bit is on a red squarebit.is_on_white()
— returnsTrue
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:
We want to have Bit follow the green squares until it gets past the last one:
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:
We want to have Bit move until it gets to 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:
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:
Since the front of bit is clear, it will do this again:
and again and again and again …
Thankfully, if you run the code, Bit will alert you if you have created an infinite loop: