-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathalgebra.py
200 lines (173 loc) · 6.87 KB
/
algebra.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
# Section 3: Algebraic simplification
# This code implements a simple computer algebra system, which takes in an
# expression made of nested sums and products, and simplifies it into a
# single sum of products. The goal is described in more detail in the
# problem set writeup.
# Much of this code is already implemented. We provide you with a
# representation for sums and products, and a top-level simplify() function
# which applies the associative law in obvious cases. For example, it
# turns both (a + (b + c)) and ((a + b) + c) into the simpler expression
# (a + b + c).
# However, the code has a gap in it: it cannot simplify expressions that are
# multiplied together. In interesting cases of this, you will need to apply
# the distributive law.
# Your goal is to fill in the do_multiply() function so that multiplication
# can be simplified as intended.
# Testing will be mathematical: If you return a flat list that
# evaluates to the same value as the original expression, you will
# get full credit.
# We've already defined the data structures that you'll use to symbolically
# represent these expressions, as two classes called Sum and Product,
# defined below. These classes both descend from the abstract Expression class.
#
# The top level function that will be called is the .simplify() method of an
# Expression.
#
# >>> expr = Sum([1, Sum([2, 3])])
# >>> expr.simplify()
# Sum([1, 2, 3])
### Expression classes _____________________________________________________
# Expressions will be represented as "Sum()" and "Product()" objects.
# These objects can be treated just like lists (they inherit from the
# "list" class), but you can test for their type using the "isinstance()"
# function. For example:
#
# >>> isinstance(Sum([1,2,3]), Sum)
# True
# >>> isinstance(Product([1,2,3]), Product)
# True
# >>> isinstance(Sum([1,2,3]), Expression) # Sums and Products are both Expressions
# True
class Expression:
"This abstract class does nothing on its own."
pass
class Sum(list, Expression):
"""
A Sum acts just like a list in almost all regards, except that this code
can tell it is a Sum using isinstance(), and we add useful methods
such as simplify().
Because of this:
* You can index into a sum like a list, as in term = sum[0].
* You can iterate over a sum with "for term in sum:".
* You can convert a sum to an ordinary list with the list() constructor:
the_list = list(the_sum)
* You can convert an ordinary list to a sum with the Sum() constructor:
the_sum = Sum(the_list)
"""
def __repr__(self):
return "Sum(%s)" % list.__repr__(self)
def simplify(self):
"""
This is the starting point for the task you need to perform. It
removes unnecessary nesting and applies the associative law.
"""
terms = self.flatten()
if len(terms) == 1:
return simplify_if_possible(terms[0])
else:
return Sum([simplify_if_possible(term) for term in terms]).flatten()
def flatten(self):
"""Simplifies nested sums."""
terms = []
for term in self:
if isinstance(term, Sum):
terms += list(term)
else:
terms.append(term)
return Sum(terms)
class Product(list, Expression):
"""
See the documentation above for Sum. A Product acts almost exactly
like a list, and can be converted to and from a list when necessary.
"""
def __repr__(self):
return "Product(%s)" % list.__repr__(self)
def simplify(self):
"""
To simplify a product, we need to multiply all its factors together
while taking things like the distributive law into account. This
method calls multiply() repeatedly, leading to the code you will
need to write.
"""
factors = []
for factor in self:
if isinstance(factor, Product):
factors += list(factor)
else:
factors.append(factor)
result = Product([1])
for factor in factors:
result = multiply(result, simplify_if_possible(factor))
return result.flatten()
def flatten(self):
"""Simplifies nested products."""
factors = []
for factor in self:
if isinstance(factor, Product):
factors += list(factor)
else:
factors.append(factor)
return Product(factors)
def simplify_if_possible(expr):
"""
A helper function that guards against trying to simplify a non-Expression.
"""
if isinstance(expr, Expression):
return expr.simplify()
else:
return expr
# You may find the following helper functions to be useful.
# "multiply" is provided for you; but you will need to write "do_multiply"
# if you would like to use it.
def multiply(expr1, expr2):
"""
This function makes sure that its arguments are represented as either a
Sum or a Product, and then passes the hard work onto do_multiply.
"""
# Simple expressions that are not sums or products can be handled
# in exactly the same way as products -- they just have one thing in them.
if not isinstance(expr1, Expression): expr1 = Product([expr1])
if not isinstance(expr2, Expression): expr2 = Product([expr2])
return do_multiply(expr1, expr2)
def do_multiply(expr1, expr2):
"""
You have two Expressions, and you need to make a simplified expression
representing their product. They are guaranteed to be of type Expression
-- that is, either Sums or Products -- by the multiply() function that
calls this one.
So, you have four cases to deal with:
* expr1 is a Sum, and expr2 is a Sum
* expr1 is a Sum, and expr2 is a Product
* expr1 is a Product, and expr2 is a Sum
* expr1 is a Product, and expr2 is a Product
You need to create Sums or Products that represent what you get by
applying the algebraic rules of multiplication to these expressions,
and simplifying.
Look above for details on the Sum and Product classes. The Python operator
'*' will not help you.
"""
# Replace this with your solution.
#Case 1:
if isinstance(expr1, Sum) and isinstance(expr2, Sum):
nt = []
for i in expr1:
for j in expr2:
nt.append(Product([i] + [j]))
print(nt)
ans = Sum(nt).simplify()
#Case 2:
if isinstance(expr1, Sum) and isinstance(expr2, Product):
nt = []
for i in expr1:
nt.append(Product([i] + expr2))
ans = Sum(nt).simplify()
#Case 3:
if isinstance(expr1, Product) and isinstance(expr2, Sum):
nt = []
for i in expr2:
nt.append(Product([i] + expr1))
ans = Sum(nt).simplify()
#Case 4:
if isinstance(expr1, Product) and isinstance(expr2, Product):
ans = Product(expr1 + expr2)
return ans