Homework 3: Tree Recursion, Sequences, Python Lists, Trees
Due by 11:59pm on Wednesday, July 10
Instructions
Download hw03.zip. Inside the archive, you will find a file called
hw03.py, along with a copy of the ok
autograder.
Submission: When you are done, submit the assignment by uploading all code files you've edited to Gradescope. You may submit more than once before the deadline; only the final submission will be scored. Check that you have successfully submitted your code on Gradescope. See Lab 0 for more instructions on submitting assignments.
Using Ok: If you have any questions about using Ok, please refer to this guide.
Readings: You might find the following references useful:
Grading: Homework is graded based on correctness. Each incorrect problem will decrease the total score by one point. This homework is out of 2 points.
Required Questions
Getting Started Videos
These videos may provide some helpful direction for tackling the coding problems on this assignment.
To see these videos, you should be logged into your berkeley.edu email.
Tree Recursion
Q1: Pascal's Triangle
Pascal's triangle gives the coefficients of a binomial expansion; if you expand the expression (a + b) ** n
,
all coefficients will be found on the n
th row of the triangle, and the coefficient of the i
th term will be at the i
th column.
Here's a part of the Pascal's trangle:
1
1 1
1 2 1
1 3 3 1
1 4 6 4 1
Every number in Pascal's triangle is defined as the sum of the item above it and the item above and to the left of it. Rows and columns are zero-indexed; that is, the first row is row 0 instead of 1 and the first column is column 0 instead of column 1. For example, the item at row 2, column 1 in Pascal's triangle is 2.
Now, define the procedure pascal(row, column)
which takes a row and a column,
and finds the value of the item at that position in Pascal's triangle.
Note that Pascal's triangle is only defined at certain areas;
use 0
if the item does not exist. For the purposes of this question,
you may also assume that row >= 0
and column >= 0
.
Hint: Pascal's Triangle can be visualized as a grid rather than a geometric, three-sided triangle. With this in mind, how does the position of the value you are trying to find relate to the positions of the two numbers that add up to it? At what coordinates of the "grid" do we find out the value and don't need further calculations? Remember that the positions are 0-indexed!
Here is a visualization of the grid:
def pascal(row, column):
"""Returns the value of the item in Pascal's Triangle
whose position is specified by row and column.
>>> pascal(0, 0) # The top left (the point of the triangle)
1
>>> pascal(0, 5) # Empty entry; outside of Pascal's Triangle
0
>>> pascal(3, 2) # Row 3 (1 3 3 1), Column 2
3
>>> pascal(4, 2) # Row 4 (1 4 6 4 1), Column 2
6
"""
"*** YOUR CODE HERE ***"
Use Ok to test your code:
python3 ok -q pascal
Sequences and Mutability
Q2: Insert Items
Write a function which takes in a list s
, a value before
, and a value
after
. It inserts after
just after each value equal to before
in s
. It
returns s
.
Important: No new lists should be created or returned.
Note: If the values passed into
before
andafter
are equal, make sure you're not creating an infinitely long list while iterating through it. If you find that your code is taking more than a few seconds to run, the function may be in an infinite loop of inserting new values.
def insert_items(s, before, after):
"""Insert after into s after each occurrence of before and then return s.
>>> test_s = [1, 5, 8, 5, 2, 3]
>>> new_s = insert_items(test_s, 5, 7)
>>> new_s
[1, 5, 7, 8, 5, 7, 2, 3]
>>> test_s
[1, 5, 7, 8, 5, 7, 2, 3]
>>> new_s is test_s
True
>>> double_s = [1, 2, 1, 2, 3, 3]
>>> double_s = insert_items(double_s, 3, 4)
>>> double_s
[1, 2, 1, 2, 3, 4, 3, 4]
>>> large_s = [1, 4, 8]
>>> large_s2 = insert_items(large_s, 4, 4)
>>> large_s2
[1, 4, 4, 8]
>>> large_s3 = insert_items(large_s2, 4, 6)
>>> large_s3
[1, 4, 6, 4, 6, 8]
>>> large_s3 is large_s
True
"""
"*** YOUR CODE HERE ***"
Use Ok to test your code:
python3 ok -q insert_items
Data Abstraction
Mobiles
Acknowledgements
This problem is based on one from Structure and Interpretation of Computer Programs Section 2.2.2.
We are making a planetarium mobile. A mobile is a type of hanging sculpture. A binary mobile consists of two arms. Each arm is a rod of a certain length, from which hangs either a planet or another mobile. For example, the below diagram shows the left and right arms of Mobile A, and what hangs at the ends of each of those arms.
We will represent a binary mobile using the data abstractions below.
- A
mobile
must have both a leftarm
and a rightarm
. - An
arm
has a positive length and must have something hanging at the end, either amobile
orplanet
. - A
planet
has a positive mass, and nothing hanging from it.
Below are the various constructors and selectors for the mobile
and arm
data abstraction.
They have already been implemented for you, though the code is not shown here.
As with any data abstraction, you should focus on what the function does rather than its specific implementation.
You are free to use any of their constructor and selector functions in the Mobiles coding exercises.
Mobile Data Abstraction (for your reference, no need to do anything here):
def mobile(left, right):
"""
Construct a mobile from a left arm and a right arm.
Arguments:
left: An arm representing the left arm of the mobile.
right: An arm representing the right arm of the mobile.
Returns:
A mobile constructed from the left and right arms.
"""
pass
def is_mobile(m):
"""
Return whether m is a mobile.
Arguments:
m: An object to be checked.
Returns:
True if m is a mobile, False otherwise.
"""
pass
def left(m):
"""
Select the left arm of a mobile.
Arguments:
m: A mobile.
Returns:
The left arm of the mobile.
"""
pass
def right(m):
"""
Select the right arm of a mobile.
Arguments:
m: A mobile.
Returns:
The right arm of the mobile.
"""
pass
Arm Data Abstraction (for your reference, no need to do anything here):
def arm(length, mobile_or_planet):
"""
Construct an arm: a length of rod with a mobile or planet at the end.
Arguments:
length: The length of the rod.
mobile_or_planet: A mobile or a planet at the end of the arm.
Returns:
An arm constructed from the given length and mobile or planet.
"""
pass
def is_arm(s):
"""
Return whether s is an arm.
Arguments:
s: An object to be checked.
Returns:
True if s is an arm, False otherwise.
"""
pass
def length(s):
"""
Select the length of an arm.
Arguments:
s: An arm.
Returns:
The length of the arm.
"""
pass
def end(s):
"""
Select the mobile or planet hanging at the end of an arm.
Arguments:
s: An arm.
Returns:
The mobile or planet at the end of the arm.
"""
pass
Q3: Mass
Implement the planet
data abstraction by completing the planet
constructor
and the mass
selector. A planet should be represented using a two-element list
where the first element is the string 'planet'
and the second element is the
planet's mass. The mass
function should return the mass of the planet
object
that is passed as a parameter.
def planet(mass):
"""Construct a planet of some mass."""
assert mass > 0
"*** YOUR CODE HERE ***"
def mass(p):
"""Select the mass of a planet."""
assert is_planet(p), 'must call mass on a planet'
"*** YOUR CODE HERE ***"
def is_planet(p):
"""Whether p is a planet."""
return type(p) == list and len(p) == 2 and p[0] == 'planet'
The total_mass
function demonstrates the use of the mobile, arm,
and planet abstractions. It has been implemented for you. You may use
the total_mass
function in the following questions.
def examples():
t = mobile(arm(1, planet(2)),
arm(2, planet(1)))
u = mobile(arm(5, planet(1)),
arm(1, mobile(arm(2, planet(3)),
arm(3, planet(2)))))
v = mobile(arm(4, t), arm(2, u))
return t, u, v
def total_mass(m):
"""Return the total mass of m, a planet or mobile.
>>> t, u, v = examples()
>>> total_mass(t)
3
>>> total_mass(u)
6
>>> total_mass(v)
9
"""
if is_planet(m):
return mass(m)
else:
assert is_mobile(m), "must get total mass of a mobile or a planet"
return total_mass(end(left(m))) + total_mass(end(right(m)))
Run the ok
tests for total_mass
to make sure that your planet
and mass
functions are implemented correctly.
Use Ok to test your code:
python3 ok -q total_mass
Q4: Balanced
Implement the balanced
function, which returns whether m
is a balanced
mobile. A mobile is balanced if both of the following conditions are met:
- The torque applied by its left arm is equal to the torque applied by its right
arm. The torque of the left arm is the length of the left rod multiplied by the
total mass hanging from that rod. Likewise for the right. For example,
if the left arm has a length of
5
, and there is amobile
hanging at the end of the left arm of total mass10
, the torque on the left side of our mobile is50
. - Each of the mobiles hanging at the end of its arms is itself balanced.
Planets themselves are balanced, as there is nothing hanging off of them.
Reminder: You may use the
total_mass
function above. Don't violate abstraction barriers. Instead, use the selector functions that have been defined.
def balanced(m):
"""Return whether m is balanced.
>>> t, u, v = examples()
>>> balanced(t)
True
>>> balanced(v)
True
>>> p = mobile(arm(3, t), arm(2, u))
>>> balanced(p)
False
>>> balanced(mobile(arm(1, v), arm(1, p)))
False
>>> balanced(mobile(arm(1, p), arm(1, v)))
False
>>> from construct_check import check
>>> # checking for abstraction barrier violations by banning indexing
>>> check(HW_SOURCE_FILE, 'balanced', ['Index'])
True
"""
"*** YOUR CODE HERE ***"
Use Ok to test your code:
python3 ok -q balanced
Trees
Q5: Finding Berries!
The squirrels on campus need your help! There are a lot of trees on campus and
the squirrels would like to know which ones contain berries. Define the function
berry_finder
, which takes in a tree and returns True
if the tree contains a
node with the value 'berry'
and False
otherwise.
Hint: To iterate through each of the branches of a particular tree, you can consider using a
for
loop to get each branch.
def berry_finder(t):
"""Returns True if t contains a node with the value 'berry' and
False otherwise.
>>> scrat = tree('berry')
>>> berry_finder(scrat)
True
>>> sproul = tree('roots', [tree('branch1', [tree('leaf'), tree('berry')]), tree('branch2')])
>>> berry_finder(sproul)
True
>>> numbers = tree(1, [tree(2), tree(3, [tree(4), tree(5)]), tree(6, [tree(7)])])
>>> berry_finder(numbers)
False
>>> t = tree(1, [tree('berry',[tree('not berry')])])
>>> berry_finder(t)
True
"""
"*** YOUR CODE HERE ***"
Use Ok to test your code:
python3 ok -q berry_finder
Q6: Maximum Path Sum
Write a function that takes in a tree and returns the maximum sum of the values along any root-to-leaf path in the tree. A root-to-leaf path is a sequence of nodes starting at the root and proceeding to some leaf of the tree. You can assume the tree will have positive numbers for its labels.
def max_path_sum(t):
"""Return the maximum root-to-leaf path sum of a tree.
>>> t = tree(1, [tree(5, [tree(1), tree(3)]), tree(10)])
>>> max_path_sum(t) # 1, 10
11
>>> t2 = tree(5, [tree(4, [tree(1), tree(3)]), tree(2, [tree(10), tree(3)])])
>>> max_path_sum(t2) # 5, 2, 10
17
"""
"*** YOUR CODE HERE ***"
Use Ok to test your code:
python3 ok -q max_path_sum
Check Your Score Locally
You can locally check your score on each question of this assignment by running
python3 ok --score
This does NOT submit the assignment! When you are satisfied with your score, submit the assignment to Gradescope to receive credit for it.
Submit Assignment
Submit this assignment by uploading any files you've edited to the appropriate Gradescope assignment. Lab 00 has detailed instructions.
Exam Practice
Homework assignments will also contain prior exam-level questions for you to take a look at. These questions have no submission component; feel free to attempt them if you'd like a challenge!
- Fall 2017 MT1 Q4a: Digital
- Summer 2018 MT1 Q5a: Won't You Be My Neighbor?
- Fall 2019 Final Q6b: Palindromes
Just For Fun Questions
The questions below are out of scope for 61A. You can try them if you want an extra challenge, but they're just puzzles that are not required for the course. Almost all students will skip them, and that's fine. We will not be prioritizing support for these questions on Ed or during Office Hours.
Q7: Towers of Hanoi
A classic puzzle called the Towers of Hanoi is a game that consists of three rods, and a number of disks of different sizes which can slide onto any rod. The puzzle starts withn
disks in a neat stack in ascending order of size on
a start
rod, the smallest at the top, forming a conical shape.
![Towers of Hanoi](https://upload.wikimedia.org/wikipedia/commons/0/07/Tower_of_Hanoi.jpeg)
end
rod,
obeying the following rules:
- Only one disk may be moved at a time.
- Each move consists of taking the top (smallest) disk from one of the rods and sliding it onto another rod, on top of the other disks that may already be present on that rod.
- No disk may be placed on top of a smaller disk.
move_stack
, which prints out the steps required to
move n
disks from the start
rod to the end
rod without violating the
rules. The provided print_move
function will print out the step to move a
single disk from the given origin
to the given destination
.
Hint: Draw out a few games with various
n
on a piece of paper and try to find a pattern of disk movements that applies to anyn
. In your solution, take the recursive leap of faith whenever you need to move any amount of disks less thann
from one rod to another. If you need more help, see the following hints.
The strategy used in Towers of Hanoi is to move all but the bottom disc to the second peg, then moving the bottom disc to the third peg, then moving all but the second disc from the second to the third peg.
One thing you don't need to worry about is collecting all the steps.
print
effectively "collects" all the results in the terminal as long as you
make sure that the moves are printed in order.
def print_move(origin, destination):
"""Print instructions to move a disk."""
print("Move the top disk from rod", origin, "to rod", destination)
def move_stack(n, start, end):
"""Print the moves required to move n disks on the start pole to the end
pole without violating the rules of Towers of Hanoi.
n -- number of disks
start -- a pole position, either 1, 2, or 3
end -- a pole position, either 1, 2, or 3
There are exactly three poles, and start and end must be different. Assume
that the start pole has at least n disks of increasing size, and the end
pole is either empty or has a top disk larger than the top n start disks.
>>> move_stack(1, 1, 3)
Move the top disk from rod 1 to rod 3
>>> move_stack(2, 1, 3)
Move the top disk from rod 1 to rod 2
Move the top disk from rod 1 to rod 3
Move the top disk from rod 2 to rod 3
>>> move_stack(3, 1, 3)
Move the top disk from rod 1 to rod 3
Move the top disk from rod 1 to rod 2
Move the top disk from rod 3 to rod 2
Move the top disk from rod 1 to rod 3
Move the top disk from rod 2 to rod 1
Move the top disk from rod 2 to rod 3
Move the top disk from rod 1 to rod 3
"""
assert 1 <= start <= 3 and 1 <= end <= 3 and start != end, "Bad start/end"
"*** YOUR CODE HERE ***"
Use Ok to test your code:
python3 ok -q move_stack
Q8: Anonymous Factorial
This question demonstrates that it's possible to write recursive functions without assigning them a name in the global frame.
The recursive factorial function can be written as a single expression by using a conditional expression.
>>> fact = lambda n: 1 if n == 1 else mul(n, fact(sub(n, 1)))
>>> fact(5)
120
However, this implementation relies on the fact (no pun intended) that
fact
has a name, to which we refer in the body of fact
. To write a
recursive function, we have always given it a name using a def
or
assignment statement so that we can refer to the function within its
own body. In this question, your job is to define fact
recursively
without giving it a name!
Write an expression that computes n
factorial using only call
expressions, conditional expressions, and lambda
expressions (no
assignment or def
statements).
Note: You are not allowed to use
make_anonymous_factorial
in your return expression.
The sub
and mul
functions from the operator
module are the only
built-in functions required to solve this problem.
from operator import sub, mul
def make_anonymous_factorial():
"""Return the value of an expression that computes factorial.
>>> make_anonymous_factorial()(5)
120
>>> from construct_check import check
>>> # ban any assignments or recursion
>>> check(HW_SOURCE_FILE, 'make_anonymous_factorial',
... ['Assign', 'AnnAssign', 'AugAssign', 'NamedExpr', 'FunctionDef', 'Recursion'])
True
"""
return 'YOUR_EXPRESSION_HERE'
Use Ok to test your code:
python3 ok -q make_anonymous_factorial