forked from yoavram/CS1001.py
-
Notifications
You must be signed in to change notification settings - Fork 0
/
matrix.py
114 lines (99 loc) · 3.62 KB
/
matrix.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
class Matrix:
def __init__(self, n_rows, m_cols, default_value=0):
assert n_rows > 0
assert m_cols > 0
assert isinstance(default_value, (int, float, complex))
self.rows = [[default_value ] * m_cols for i in range(n_rows)]
def dim(self):
'''return tuple -> num of rows, num of cols'''
return (len(self.rows), len(self.rows[0]))
def __repr__(self):
return 'Matrix: %d rows, %d cols\n' % self.dim() + str(self.rows[0])
def __getitem__(self, ij):
'''ij is a tuple (i,j). Allows m[i,j] instead m[i][j]'''
i,j = ij
if isinstance(i, int) and isinstance(j, int):
return self.rows[i][j]
elif isinstance(i, slice) and isinstance(j, slice):
M = Matrix(1,1) # to be overwritten
M.rows = [row[j] for row in self.rows[i]]
return M
else:
return NotImplemented
def __setitem__(self, ij, val):
'''ij is a tuple (i,j). Allows m[i,j] instead m[i][j]'''
i,j = ij
if isinstance(i,int) and isinstance(j,int):
assert isinstance(val, (int, float, complex))
self.rows[i][j] = val
elif isinstance(i,slice) and isinstance(j,slice):
assert isinstance(val, Matrix)
n,m = val.dim()
s_rows = self.rows[i]
assert len(s_rows) == n and len(s_rows[0][j]) == m
for s_row, v_row in zip(s_rows,val.rows):
s_row[j] = v_row
else:
return NotImplemented
def __eq__(self, other):
assert isinstance(other, Matrix)
if self.dim() != other.dim():
return False
n,m = self.dim()
for i in range(n):
for j in range(m):
if self[i,j] != other[i,j]:
return False
return True
def __add__(self, other):
return self._entrywise_op(other, lambda x,y: x + y)
def __sub__(self, other):
return self._entrywise_op(other, lambda x,y: x - y)
def __neg__(self):
n,m = self.dim()
return Matrix(n, m, 0) - self
def _entrywise_op(self, other, op):
assert isinstance(other, Matrix)
assert self.dim() == other.dim()
n,m = self.dim()
M = Matrix(n, m)
for i in range(n):
for j in range(m):
M[i,j] = op(self[i,j], other[i,j])
return M
def __mul__(self, other):
'''multilpy by scalar or another matrix'''
if isinstance(other, (int,float,complex)):
n,m = self.dim()
M = Matrix(n, m, other)
return self._entrywise_op(M, lambda x,y: x * y)
elif isinstance(other, Matrix):
n1,m1 = self.dim()
n2,m2 = other.dim()
assert m1 == n2
M = Matrix(n1,m2)
for i in range(n1):
for j in range(m2):
M[i,j] = sum([self[i,k] * other[k,j] for k in range(m1)])
return M
else:
return NotImplemented
__rmul__ = __mul__
def prettyprint(self):
return str.join('\n',[str(row) for row in self.rows])
def save(self, filename):
'''save to file'''
with open(filename, "w") as fout:
fout.write(self.prettyprint())
@staticmethod
def load(filename):
'''load from file'''
rows = []
with open(filename, 'r') as fin:
for line in fin:
line = line.strip()
row = eval(line)
rows.append(row)
M = Matrix(1,1)
M.rows = rows
return M