-
Notifications
You must be signed in to change notification settings - Fork 39
/
Copy pathdaily_coding_problem_36_40.py
159 lines (129 loc) · 6 KB
/
daily_coding_problem_36_40.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
import math
def coding_problem_36(tree):
"""
Given the root to a binary search tree, find the second largest node in the tree.
Example:
>>> tree = [9, [4, [1, None, None], [7, [5, None, None], None]], [31, [14, None, None], [82, None, None]]]
>>> coding_problem_36(tree)
31
Note: in a binary search tree, the second largest node is the root of the largest element.
"""
value, _, right = tree
parent = value
while right:
parent = value
value, _, right = right
return parent
def coding_problem_37(s):
"""
The power set of a set is the set of all its subsets. Write a function that, given a set, generates its power set.
You may also use a list or array to represent a set. Example:
>>> sorted([sorted(subset) for subset in coding_problem_37({1, 2, 3})], key=len)
[[], [1], [2], [3], [1, 2], [1, 3], [2, 3], [1, 2, 3]]
"""
power_set = [[]]
for element in s:
power_set.extend([[element] + subset for subset in power_set])
return power_set
def coding_problem_38(n):
"""
You have an N by N board. Write a function that, given N, returns the number of possible arrangements of the board
where N queens can be placed on the board without threatening each other, i.e. no two queens share the same row,
column, or diagonal. From this Wikipedia article https://en.wikipedia.org/wiki/Eight_queens_puzzle :
>>> [coding_problem_38(n + 1) for n in range(7)]
[1, 0, 0, 2, 10, 4, 40]
Note: this could be made much faster by passing only valid coordinates instead of all of them each time, but
it is an exercise left for the reader ;)
"""
def is_threat(new_queen, board_arrangement):
for already_placed in board_arrangement:
if new_queen[0] == already_placed[0] or new_queen[1] == already_placed[1] or \
abs(new_queen[0] - already_placed[0]) == abs(new_queen[1] - already_placed[1]):
return True
return False # otherwise
all_coordinates = [(i, j) for i in range(n) for j in range(n)]
valid_arrangements = {frozenset()} # set for duplicate removal, empty initial arrangement
for _ in range(n):
valid_arrangements = set(arrangement.union([coord]) for arrangement in valid_arrangements
for coord in all_coordinates if not is_threat(coord, arrangement))
return len(valid_arrangements)
def coding_problem_39(cells):
"""
Conway's Game of Life takes place on an infinite two-dimensional board of square cells. Each cell is either dead or
alive, and at each tick, the following rules apply:
Any live cell with less than two live neighbours dies.
Any live cell with two or three live neighbours remains living.
Any live cell with more than three live neighbours dies.
Any dead cell with exactly three live neighbours becomes a live cell.
A cell neighbours another cell if it is horizontally, vertically, or diagonally adjacent.
Implement Conway's Game of Life. It should be able to be initialized with a starting list of live cell coordinates
and the number of steps it should run for. Once initialized, it should print out the board state at each step.
Since it's an infinite board, print out only the relevant coordinates, i.e. from the top-leftmost live cell to
bottom-rightmost live cell.
You can represent a live cell with an asterisk (*) and a dead cell with a dot (.).
Example: simulate a glider (https://en.wikipedia.org/wiki/Glider_(Conway%27s_Life)) advancing one step.
>>> gol = coding_problem_39(((0, 1), (1, 2), (2, 0), (2, 1), (2, 2)))
>>> for _ in range(5):
... print(gol)
... gol.simulate()
..*
*.*
.**
<BLANKLINE>
*..
.**
**.
<BLANKLINE>
.*.
..*
***
<BLANKLINE>
*.*
.**
.*.
<BLANKLINE>
..*
*.*
.**
<BLANKLINE>
"""
class GameOfLife(object):
def __init__(self, cells):
self.displacements = tuple((xd, yd) for yd in [-1, 0, 1] for xd in [-1, 0, 1] if xd != 0 or yd != 0)
self.cells = set(cells)
def __str__(self):
xmin, xmax, ymin, ymax = [min_max(xy) for xy in zip(*self.cells) for min_max in [min, max]]
return ''.join(('*' if (x, y) in self.cells else '.') + ('\n' if x == xmax else '')
for y in range(ymin, ymax + 1) for x in range(xmin, xmax + 1))
def alive_next_round(self, x, y):
neighbour_count = sum(1 if (x + xd, y + yd) in self.cells else 0 for xd, yd in self.displacements)
return neighbour_count == 3 or ((x, y) in self.cells and neighbour_count == 2)
def simulate(self, steps=1):
for _ in range(steps):
active_cells = set((x + xd, y + yd) for x, y in self.cells for xd in [-1, 0, 1] for yd in [-1, 0, 1])
self.cells = set((x, y) for x, y in active_cells if self.alive_next_round(x, y))
return GameOfLife(cells)
def coding_problem_40(numbers):
"""
Given an array of integers where every integer occurs three times except for one integer, which only occurs once,
find and return the non-duplicated integer. Do this in O(N) time and O(1) space.
Examples:
>>> coding_problem_40([6, 1, 3, 3, 3, 6, 6])
1
>>> coding_problem_40([13, 19, 13, 13])
19
Note: the code below fails for negative numbers, but you could shift the entire array by the minimum.
Note2: the implementation below makes it evident that the space complexity is O(log_2(n)) rather than O(1).
This is always the case, unless a specific integer type or a maximum value are specified (they are not above).
"""
bits = [0] * int(math.ceil(math.log(max(numbers), 2))) # low endian
for number in numbers:
index = 0
while number > 0:
bits[index] += number & 1
number //= 2
index += 1
return sum((b % 3) * 2**i for i, b in enumerate(bits))
if __name__ == '__main__':
import doctest
doctest.testmod(verbose=True)