Skip to content

Commit 006cefd

Browse files
guillaume-chevalierkeon
authored andcommitted
Add convolve iteration algorithm with tests (#428)
* added convolve algorithm with tests * Fixed tox.ini * Removed dependency
1 parent e1eb4bd commit 006cefd

11 files changed

+334
-1
lines changed

README.md

+2
Original file line numberDiff line numberDiff line change
@@ -159,6 +159,8 @@ If you want to uninstall algorithms, it is as simple as:
159159
- [skyline](algorithms/heap/skyline.py)
160160
- [sliding_window_max](algorithms/heap/sliding_window_max.py)
161161
- [binary_heap](algorithms/heap/binary_heap.py)
162+
- [iterables](algorithms/iterables)
163+
- [convolved](algorithms/iterables/convolved.py)
162164
- [k_closest_points](algorithms/heap/k_closest_points.py)
163165
- [linkedlist](algorithms/linkedlist)
164166
- [add_two_numbers](algorithms/linkedlist/add_two_numbers.py)

README_CN.md

+2
Original file line numberDiff line numberDiff line change
@@ -157,6 +157,8 @@ pip3 uninstall -y algorithms
157157
- [merge_sorted_k_lists:合并k个有序链](algorithms/heap/merge_sorted_k_lists.py)
158158
- [skyline:天际线](algorithms/heap/skyline.py)
159159
- [sliding_window_max:滑动窗口最大值](algorithms/heap/sliding_window_max.py)
160+
- [iterables : 迭代](algorithms/iterables)
161+
- [convolved : 卷积](algorithms/iterables/convolved.py)
160162
- [linkedlist:链表](algorithms/linkedlist)
161163
- [add_two_numbers:链表数相加](algorithms/linkedlist/add_two_numbers.py)
162164
- [copy_random_pointer:复制带有随机指针的链表](algorithms/linkedlist/copy_random_pointer.py)

README_FR.md

+2
Original file line numberDiff line numberDiff line change
@@ -158,6 +158,8 @@ Si vous voulez désinstaller le paquet algorithms, il suffit de procéder comme
158158
- [skyline](algorithms/heap/skyline.py)
159159
- [sliding_window_max](algorithms/heap/sliding_window_max.py)
160160
- [binary_heap](algorithms/heap/binary_heap.py)
161+
- [iterables](algorithms/iterables)
162+
- [convolved](algorithms/iterables/convolved.py)
161163
- [linkedlist](algorithms/linkedlist)
162164
- [add_two_numbers](algorithms/linkedlist/add_two_numbers.py)
163165
- [copy_random_pointer](algorithms/linkedlist/copy_random_pointer.py)

README_GE.md

+2
Original file line numberDiff line numberDiff line change
@@ -165,6 +165,8 @@ Um das Projekt zu deinstallieren tippen Sie folgendes:
165165
- [skyline](algorithms/heap/skyline.py)
166166
- [sliding_window_max](algorithms/heap/sliding_window_max.py)
167167
- [binary_heap](algorithms/heap/binary_heap.py)
168+
- [iterables](algorithms/iterables)
169+
- [convolved](algorithms/iterables/convolved.py)
168170
- [linkedlist](algorithms/linkedlist)
169171
- [add_two_numbers](algorithms/linkedlist/add_two_numbers.py)
170172
- [copy_random_pointer](algorithms/linkedlist/copy_random_pointer.py)

README_JP.md

+2
Original file line numberDiff line numberDiff line change
@@ -159,6 +159,8 @@ if __name__ == "__main__":
159159
- [skyline](algorithms/heap/skyline.py)
160160
- [sliding_window_max](algorithms/heap/sliding_window_max.py)
161161
- [binary_heap](algorithms/heap/binary_heap.py)
162+
- [iterables : 繰り返し可能な](algorithms/iterables)
163+
- [convolved](algorithms/iterables/convolved.py)
162164
- [linkedlist : 連結リスト](algorithms/linkedlist)
163165
- [add_two_numbers](algorithms/linkedlist/add_two_numbers.py)
164166
- [copy_random_pointer](algorithms/linkedlist/copy_random_pointer.py)

README_KR.md

+2
Original file line numberDiff line numberDiff line change
@@ -155,6 +155,8 @@ if __name__ == "__main__":
155155
- [skyline](algorithms/heap/skyline.py)
156156
- [sliding_window_max](algorithms/heap/sliding_window_max.py)
157157
- [binary_heap](algorithms/heap/binary_heap.py)
158+
- [iterable : 반복 가능한 것](algorithms/iterables)
159+
- [convolved](algorithms/iterables/convolved.py)
158160
- [linkedlist : 연결 리스트](algorithms/linkedlist)
159161
- [add_two_numbers](algorithms/linkedlist/add_two_numbers.py)
160162
- [copy_random_pointer](algorithms/linkedlist/copy_random_pointer.py)

README_PTBR.md

+2
Original file line numberDiff line numberDiff line change
@@ -157,6 +157,8 @@ Se você deseja desinstalar os algoritmos, é tão simples quanto:
157157
- [skyline](algorithms/heap/skyline.py)
158158
- [sliding_window_max](algorithms/heap/sliding_window_max.py)
159159
- [binary_heap](algorithms/heap/binary_heap.py)
160+
- [iterables](algorithms/iterables)
161+
- [convolved](algorithms/iterables/convolved.py)
160162
- [linkedlist](algorithms/linkedlist)
161163
- [add_two_numbers](algorithms/linkedlist/add_two_numbers.py)
162164
- [copy_random_pointer](algorithms/linkedlist/copy_random_pointer.py)

algorithms/iterables/__init__.py

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
from .convolved import *

algorithms/iterables/convolved.py

+154
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,154 @@
1+
"""Iterable to get every convolution window per loop iteration.
2+
3+
## Example Usage
4+
5+
```
6+
from algorithms.iterables import convolved
7+
# This would also work: from conv import convolved
8+
9+
some_list = [1, 2, 3]
10+
for kernel_hover in convolved(some_list, kernel_size=2, stride=1, padding=2, default_value=42):
11+
print(kernel_hover)
12+
```
13+
14+
## Result:
15+
16+
```
17+
[42, 42]
18+
[42, 1]
19+
[1, 2]
20+
[2, 3]
21+
[3, 42]
22+
[42, 42]
23+
```
24+
25+
"""
26+
27+
28+
def convolved(iterable, kernel_size=1, stride=1, padding=0, default_value=None):
29+
"""Iterable to get every convolution window per loop iteration.
30+
31+
For example:
32+
`convolved([1, 2, 3, 4], kernel_size=2)`
33+
will produce the following result:
34+
`[[1, 2], [2, 3], [3, 4]]`.
35+
`convolved([1, 2, 3], kernel_size=2, stride=1, padding=2, default_value=42)`
36+
will produce the following result:
37+
`[[42, 42], [42, 1], [1, 2], [2, 3], [3, 42], [42, 42]]`
38+
39+
Arguments:
40+
iterable: An object to iterate on. It should support slice indexing if `padding == 0`.
41+
kernel_size: The number of items yielded at every iteration.
42+
stride: The step size between each iteration.
43+
padding: Padding must be an integer or a string with value `SAME` or `VALID`. If it is an integer, it represents
44+
how many values we add with `default_value` on the borders. If it is a string, `SAME` means that the
45+
convolution will add some padding according to the kernel_size, and `VALID` is the same as
46+
specifying `padding=0`.
47+
default_value: Default fill value for padding and values outside iteration range.
48+
49+
For more information, refer to:
50+
- https://github.com/guillaume-chevalier/python-conv-lib/blob/master/conv/conv.py
51+
- https://github.com/guillaume-chevalier/python-conv-lib
52+
- MIT License, Copyright (c) 2018 Guillaume Chevalier
53+
"""
54+
# Input validation and error messages
55+
if not hasattr(iterable, '__iter__'):
56+
raise ValueError(
57+
"Can't iterate on object.".format(
58+
iterable))
59+
if stride < 1:
60+
raise ValueError(
61+
"Stride must be of at least one. Got `stride={}`.".format(
62+
stride))
63+
if not (padding in ['SAME', 'VALID'] or type(padding) in [int]):
64+
raise ValueError(
65+
"Padding must be an integer or a string with value `SAME` or `VALID`.")
66+
if not isinstance(padding, str):
67+
if padding < 0:
68+
raise ValueError(
69+
"Padding must be of at least zero. Got `padding={}`.".format(
70+
padding))
71+
else:
72+
if padding == 'SAME':
73+
padding = kernel_size // 2
74+
elif padding == 'VALID':
75+
padding = 0
76+
if not type(iterable) == list:
77+
iterable = list(iterable)
78+
79+
# Add padding to iterable
80+
if padding > 0:
81+
pad = [default_value] * padding
82+
iterable = pad + list(iterable) + pad
83+
84+
# Fill missing value to the right
85+
remainder = (kernel_size - len(iterable)) % stride
86+
extra_pad = [default_value] * remainder
87+
iterable = iterable + extra_pad
88+
89+
i = 0
90+
while True:
91+
if i > len(iterable) - kernel_size:
92+
break
93+
yield iterable[i:i + kernel_size]
94+
i += stride
95+
96+
def convolved_1d(iterable, kernel_size=1, stride=1, padding=0, default_value=None):
97+
"""1D Iterable to get every convolution window per loop iteration.
98+
99+
For more information, refer to:
100+
- https://github.com/guillaume-chevalier/python-conv-lib/blob/master/conv/conv.py
101+
- https://github.com/guillaume-chevalier/python-conv-lib
102+
- MIT License, Copyright (c) 2018 Guillaume Chevalier
103+
"""
104+
return convolved(iterable, kernel_size, stride, padding, default_value)
105+
106+
107+
def convolved_2d(iterable, kernel_size=1, stride=1, padding=0, default_value=None):
108+
"""2D Iterable to get every convolution window per loop iteration.
109+
110+
For more information, refer to:
111+
- https://github.com/guillaume-chevalier/python-conv-lib/blob/master/conv/conv.py
112+
- https://github.com/guillaume-chevalier/python-conv-lib
113+
- MIT License, Copyright (c) 2018 Guillaume Chevalier
114+
"""
115+
kernel_size = dimensionize(kernel_size, nd=2)
116+
stride = dimensionize(stride, nd=2)
117+
padding = dimensionize(padding, nd=2)
118+
119+
for row_packet in convolved(iterable, kernel_size[0], stride[0], padding[0], default_value):
120+
transposed_inner = []
121+
for col in tuple(row_packet):
122+
transposed_inner.append(list(
123+
convolved(col, kernel_size[1], stride[1], padding[1], default_value)
124+
))
125+
126+
if len(transposed_inner) > 0:
127+
for col_i in range(len(transposed_inner[0])):
128+
yield tuple(row_j[col_i] for row_j in transposed_inner)
129+
130+
131+
def dimensionize(maybe_a_list, nd=2):
132+
"""Convert integers to a list of integers to fit the number of dimensions if
133+
the argument is not already a list.
134+
135+
For example:
136+
`dimensionize(3, nd=2)`
137+
will produce the following result:
138+
`(3, 3)`.
139+
`dimensionize([3, 1], nd=2)`
140+
will produce the following result:
141+
`[3, 1]`.
142+
143+
For more information, refer to:
144+
- https://github.com/guillaume-chevalier/python-conv-lib/blob/master/conv/conv.py
145+
- https://github.com/guillaume-chevalier/python-conv-lib
146+
- MIT License, Copyright (c) 2018 Guillaume Chevalier
147+
"""
148+
if not hasattr(maybe_a_list, '__iter__'):
149+
# Argument is probably an integer so we map it to a list of size `nd`.
150+
now_a_list = [maybe_a_list] * nd
151+
return now_a_list
152+
else:
153+
# Argument is probably an `nd`-sized list.
154+
return maybe_a_list

tests/test_iterables.py

+161
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,161 @@
1+
from algorithms.iterables import (
2+
convolved_1d, convolved_2d
3+
)
4+
5+
import itertools
6+
import unittest
7+
8+
9+
class TestConvolved1D(unittest.TestCase):
10+
"""Tests copied from:
11+
- https://github.com/guillaume-chevalier/python-conv-lib/blob/master/conv/tests/test_convolved_1d.py
12+
- MIT License, Copyright (c) 2018 Guillaume Chevalier
13+
"""
14+
15+
def test_trivial_loop(self):
16+
expected = tuple(range(7))
17+
result = []
18+
19+
for kernel_hover in convolved_1d(expected, kernel_size=1, padding=0, stride=1):
20+
result.append(*kernel_hover)
21+
result = tuple(result)
22+
23+
self.assertEqual(expected, result)
24+
25+
def test_seven_for_with_stride_two(self):
26+
expected = tuple(range(0, 7, 2))
27+
result = []
28+
29+
for kernel_hover in convolved_1d(list(range(7)), kernel_size=1, padding=0, stride=2):
30+
result.append(*kernel_hover)
31+
result = tuple(result)
32+
33+
self.assertEqual(expected, result)
34+
35+
def test_seven_for_with_padding_one(self):
36+
expected = tuple([42] + list(range(0, 7)) + [42])
37+
result = []
38+
39+
for kernel_hover in convolved_1d(list(range(7)), kernel_size=1, padding=1, stride=1, default_value=42):
40+
result.append(*kernel_hover)
41+
result = tuple(result)
42+
43+
self.assertEqual(expected, result)
44+
45+
def test_seven_kernel_of_two(self):
46+
expected = tuple([a, b] for a, b in zip(list(range(0, 6)), list(range(1, 7))))
47+
result = []
48+
49+
for kernel_hover in convolved_1d(list(range(7)), kernel_size=2, padding=0, stride=1):
50+
result.append(kernel_hover)
51+
result = tuple(result)
52+
53+
self.assertEqual(expected, result)
54+
55+
def test_seven_kernel_of_two_and_stride_two(self):
56+
expected = ([0, 1], [2, 3], [4, 5], [6, None])
57+
result = []
58+
59+
for kernel_hover in convolved_1d(list(range(7)), kernel_size=2, padding=0, stride=2):
60+
result.append(kernel_hover)
61+
result = tuple(result)
62+
63+
self.assertEqual(expected, result)
64+
65+
def test_seven_kernel_of_two_and_stride_two_and_padding_two(self):
66+
expected = ([None, None], [0, 1], [2, 3], [4, 5], [6, None], [None, None])
67+
result = []
68+
69+
for kernel_hover in convolved_1d(list(range(7)), kernel_size=2, padding=2, stride=2):
70+
result.append(kernel_hover)
71+
result = tuple(result)
72+
73+
self.assertEqual(expected, result)
74+
75+
def test_seven_kernel_of_two_and_stride_two_and_padding_three(self):
76+
expected = ([None, None], [None, 0], [1, 2], [3, 4], [5, 6], [None, None], [None, None])
77+
result = []
78+
79+
for kernel_hover in convolved_1d(list(range(7)), kernel_size=2, padding=3, stride=2):
80+
result.append(kernel_hover)
81+
result = tuple(result)
82+
83+
self.assertEqual(expected, result)
84+
85+
def test_seven_kernel_of_three_and_stride_two_and_padding_two(self):
86+
expected = ([None, None, 0], [0, 1, 2], [2, 3, 4], [4, 5, 6], [6, None, None])
87+
result = []
88+
89+
for kernel_hover in convolved_1d(list(range(7)), kernel_size=3, padding=2, stride=2):
90+
result.append(kernel_hover)
91+
result = tuple(result)
92+
93+
self.assertEqual(expected, result)
94+
95+
def test_seven_kernel_of_two_and_stride_three_and_padding_two(self):
96+
expected = ([None, None], [1, 2], [4, 5], [None, None])
97+
result = []
98+
99+
for kernel_hover in convolved_1d(list(range(7)), kernel_size=2, padding=2, stride=3):
100+
result.append(kernel_hover)
101+
result = tuple(result)
102+
103+
self.assertEqual(expected, result)
104+
105+
def test_seven_kernel_of_three_and_stride_three_and_padding_three(self):
106+
expected = ([None, None, None], [0, 1, 2], [3, 4, 5], [6, None, None], [None, None, None])
107+
result = []
108+
109+
for kernel_hover in convolved_1d(list(range(7)), kernel_size=3, padding=3, stride=3):
110+
result.append(kernel_hover)
111+
result = tuple(result)
112+
113+
self.assertEqual(expected, result)
114+
115+
116+
class TestConvolved2D(unittest.TestCase):
117+
"""Tests copied from:
118+
- https://github.com/guillaume-chevalier/python-conv-lib/blob/master/conv/tests/test_convolved_1d.py
119+
- MIT License, Copyright (c) 2018 Guillaume Chevalier
120+
"""
121+
122+
def test_trivial_1x1_loop(self):
123+
base = tuple(tuple(range(i ** 2, 7 + i ** 2)) for i in range(7))
124+
expected = tuple(([i],) for i in itertools.chain(*base))
125+
result = []
126+
127+
for kernel_hover in convolved_2d(base, kernel_size=1, padding=0, stride=1):
128+
result.append(tuple(kernel_hover))
129+
result = tuple(result)
130+
131+
self.assertEqual(expected, result)
132+
133+
def test_simple_2x2_loop_on_3x2(self):
134+
base = [
135+
[1, 2],
136+
[3, 4],
137+
[5, 6]
138+
]
139+
expected = (
140+
(
141+
[1, 2],
142+
[3, 4]
143+
), (
144+
[3, 4],
145+
[5, 6]
146+
)
147+
)
148+
result = []
149+
150+
for kernel_hover in convolved_2d(base, kernel_size=2, padding=0, stride=1):
151+
result.append(tuple(kernel_hover))
152+
result = tuple(result)
153+
154+
print(result)
155+
print(expected)
156+
157+
self.assertEqual(expected, result)
158+
159+
160+
if __name__ == "__main__":
161+
unittest.main()

tree.md

+4-1
Original file line numberDiff line numberDiff line change
@@ -107,6 +107,9 @@ algorithms
107107
│   ├── skyline.py
108108
│   └── sliding_window_max.py
109109
├── __init__.py
110+
├── iterables
111+
│   ├── __init__.py
112+
│   └── convolved.py
110113
├── linkedlist
111114
│   ├── add_two_numbers.py
112115
│   ├── copy_random_pointer.py
@@ -296,5 +299,5 @@ algorithms
296299
└── union-find
297300
└── count_islands.py
298301
299-
27 directories, 267 files
302+
28 directories, 269 files
300303
```

0 commit comments

Comments
 (0)