HW 1 is made of many Bit problems.
Part-a is the first set of problems on this page. Part-b depends on lecture-3, and we'll release it at that time. The whole thing is due Wed Jan 15th at 11:55pm. The mechanics of submitting your work will be explained in the part-b text.
Lair helper hours (linked off course page) begin for the quarter on Sun night. The experimental server saves your code each time you click the Run button. There is also a (seldom used) "save" command in a menu, if you want to save your code without running it.
Solve these problems using loops and logic like the lecture examples (not using Python features like for, range, int counting, break, and, or .. which we have not covered yet.)
Work on your code using one case at a time. As a last step, select the "Run All" option in the menu and Run to check your code against all the cases. If successful, the output is "All Correct" at the top with the big checkmark, which means your code produces the correct output. You should also check your code over for clean style (also covered Fri).
The part-a problems are 1 function each - no extra decomposition.
1. top-rgb
2. top-bot
3. top-logic1
4. top-logic2
5. change-green
Note for problems 6-7-8-9 and later: When bit moves across a series of squares in a straight line, we'll call that a "line move". Many of the problems below involve a series a line moves, e.g. Bit follows a horizontal line, then turns and follows a vertical line, and then turns and follows another horizontal line. In your code, solve each line move with its own while loop. In particular, where bit stops for each line move should be encoded in the test at the top of each loop. This is a good way to code a series of line moves, and it's the way we want you to do these. When the problem statement says "use three loops", that is a reminder of this rule; it does not limit your use of if-statements, just spelling out how many loops there should be.
6. hidey-hole
Suggestion, as above: One loop to move right, with its test stopping at the right spot. Then a second loop to go down.
7. cave-blue
8. downward-red
9. upward-blue
10. up-green
Suggestion: write one while loop that gets to the top first and test that. Then add if-logic inside the loop to detect and handle the the niches on the way. See the lecture-2 "dip" example.
Before
After
11. tunnel
Before
After
The code for tunnel is a little repetitious. For part-b, we'll see a technique to tighten that up.
12. spiral escape
Bit is stuck in a terrible maze. Use 5 loops to get bit to the exit square on the left edge like this: 1. Move bit forward until a solid black square appears to bit's left. 2. Turn right and then move forward until the first green square. 3. Turn right and move one square forward, into the 1-wide vertical hole. Move down the hole until an empty square appears to bit's right. 4. Turn right and then move until blocked. 5. Turn right and then move until on a non-blue square. 6. Turn left and move exactly one square - this is the exit. Paint it green. Bit praises your coding genius from the exit square!
Suggestion - this problem has, if we're being honest, every line move situation we could think of. As explained above, each line move should get its own while-loop. You want to use a drawing to work out the geometry and logic for the loops, getting bit to stop on the right square using each while-test. Here is a drawing based on on the Case-1 maze, showing how you might focus on the 5 stopping squares to work out the while-tests. This is good example of using a drawing to work through a complex sequence. Don't hesitate to click Run on your partially entered code — a handy way to see what's working and what's next.
Stop Here - you are done with part-a of the homework. We will release part-b on Friday, and also instructions for turning in your code (which the experimental server saves on each click of the Run button). Part-b will have a few problems, using techniques from Friday's lecture.
These problems use the ideas from Friday lecture.
This problem deals with the double-move issue of how to do 2 moves within a loop. This is a single-function problem.
Starting with this problem, there will be one or two helper functions per problem. Recall the key theme of lecture-3 — call the helper function. There are radio-buttons at the top of the page on the experimental server that scroll to each function and adjust which case is tested, or you can do it manually. All the function defs should be at the leftmost indent in the text; not one function indented inside another. The line "bit = Bit(filename)" should only appear once in the whole program (the Fri lecture examples demonstrate this). The Python "pass" construct is a placeholder showing where you code goes, and it should be deleted.
For these problems, we are specifying the function pre/post conditions here in the handout instead of in the triple-quote """Pydoc""" in the code.
a. The helper function do_tree() draws a small tree made of 4 painted squares. The bottom of the tree is made of a green square, a red square, and a green square in a row. Then above the red square is one more green square. Before the function runs, bit is one square away and facing what will be the red square. The function draws the tree with a series of moves, turns, and paints, but no loops. It's ok if bit moves a little inefficiently, so long as the tree is drawn. At the end, leave bit on the original square, but facing the opposite direction from the beginning. Assume there is enough clear space to draw the tree.
Before do_tree()
After do_tree()
b. For the do_forrest() function, bit starts facing the right side of the world. Move bit straight forward until blocked. For each moved-to square, if the square is red, draw a tree just above the red square.
Before do_forest()
After do_forest()
a. For this problem, the do_room() helper function paints one room green. Each room is 1 square high and 1 or more squares wide. There is one empty square - a door - that is the opening into the room. Bit begins in the door square, facing into the room. Go into the room and paint all its squares green. Then return to occupy the door square, facing out of the room. It's ok if bit goes some extra distance, so long as all the painting is done and bit gets back to the door. Try to only paint each square green a single time. Our solution uses 3 while loops which is a good approach — get to one side, paint all the squares green, find the exit.
b. Bit starts at the beginning of a 1-wide hall. Move bit forward to the end of the hall. For each moved to square, a door to a room may appear at the left or right, or both. Fill all the rooms with green. End with bit at end end of the hall, facing the original direction, like this:
This problem is an algorithmic classic. It's helpful to be familiar with the Fill example from lecture-3 which has some of the same issues as Checkerboard. For the Checkerboard problem, bit starts in the upper left corner, facing down. Move bit to the lower left corner, filling in all the rows with a checkerboard pattern, like this:
Before:
After:
It's handy to have terms for the two row types — we'll say the topmost row is "type 0", since the blue squares begin right away, and the second row is "type 1".
Your code should be able to solve any size world that is 2x2 or larger, and the world may not be square. The double-move issue appears a couple times in the checkerboard, so keep your solution to the blue-green problem above in mind.
We will decompose out two helpers, solving each type of row — do_type0() and do_type1().
There is a cute coding trick that helps with the code for the type-0 and type-1 rows. It's quite non-obvious, so we're going to nudge you in the right direction: inside your do_type1() function, you can call the do_type0() function to do almost all the work. As a result, the do_type1() function will be short and will not itself contain any loops.
Get the helpers working first, before solving the whole checkerboard. Cases 1-6 in the menu test the helpers, testing them with both even and odd width rows.
The pre/post for the helpers are provided in the """Pydoc""" on each function.
def do_row0(bit): """ Begin facing down in the row to fill. Move towards the right side of the world, making a type-0 row. End at the left side of the row, facing down. """ pass def do_row1(bit): """ Begin facing down in the row to fill. Move towards the right side of the world, making a type-1 row. End at the left side of the row, facing down. """ pass
Type-0 before/after:
Type-1 before/after:
Now the main challenge — write code in do_checkerboard() to solve the whole thing. Bit begins in the upper left corner facing down, and ends in the lower left corner facing down. Use the same structure as the fill() example: in the loop, move into a row and then solve it. Handle the first row separately.
You have earned a moment of satisfaction when it finally works.
Below is the link to turn in your code to "Paperless", the system where the section leaders review and grade your code. The Experimental server is storing your code each time you click the Run button, as you can tell if you close a tab and then re-open it. Use the page linked below to review and send your from the experimental server to paperless. Your code needs to have passed the Run All case, getting the big checkmark, and the code should be cleaned to follow proper PEP8 Python coding style as explained here: Python Style 1
And you're done - welcome to Python coding!