-
Notifications
You must be signed in to change notification settings - Fork 163
/
Copy path06_MaxProductOfThree.py
131 lines (96 loc) · 3.38 KB
/
06_MaxProductOfThree.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
"""
# MaxProductOfThree
Maximize A[P] * A[Q] * A[R] for any triplet (P, Q, R)
https://codility.com/programmers/task/max_product_of_three/
## Problem Description
A non-empty array A consisting of N integers is given.
The product of triplet (P, Q, R) equates to A[P] * A[Q] * A[R] (0 ≤ P < Q < R < N).
For example, array A such that:
A[0] = -3
A[1] = 1
A[2] = 2
A[3] = -2
A[4] = 5
A[5] = 6
contains the following example triplets:
(0, 1, 2), product is −3 * 1 * 2 = −6
(1, 2, 4), product is 1 * 2 * 5 = 10
(2, 4, 5), product is 2 * 5 * 6 = 60
Your goal is to find the maximal product of any triplet.
Write a function:
def solution(A)
that, given a non-empty array A, returns the value of the maximal product of any triplet.
For example, given array A such that:
A[0] = -3
A[1] = 1
A[2] = 2
A[3] = -2
A[4] = 5
A[5] = 6
the function should return 60, as the product of triplet (2, 4, 5) is maximal.
Write an efficient algorithm for the following assumptions:
N is an integer within the range [3..100,000];
each element of array A is an integer within the range [−1,000..1,000].
Copyright 2009–2022 by Codility Limited. All Rights Reserved.
Unauthorized copying, publication or disclosure prohibited.
----------------
## Notes
Sorting the array then multiplying the three biggest numbers will only get you a score of 44%.
The 'trick' is that two negatives make a positive.
eg: [-100, -100, -100, 0, 50, 50, 50]
The product of the top 3 is 125000,
but the product of the bottom two (-100, 100) with the top one (50) is 500000!
## Solution O(N*log(N))
1. Sort them
2. Return the larger product of
* the lowest 2 numbers and the highest number, or
* the highest 3 numbers
https://app.codility.com/demo/results/training4Z42TU-R8E/
"""
import unittest
import random
def solution(A):
"""
:param A: list[int]
:return: [int]
"""
A.sort()
return max(
A[0] * A[1] * A[-1],
A[-3] * A[-2] * A[-1]
)
RANGE_A = (-1000, 1000)
RANGE_N = (3, 100000)
class TestExercise(unittest.TestCase):
"""
example: example test
one_triple: three elements
simple1: simple tests
simple2: simple tests
small_random: random small, length=100
medium_range: -1000, -999,...,1000, length 1000 = 999000000
medium_random: length ~10000
large_random: random large, length 100000
large_range: 2000 * (-10..10) + [-1000, 500, -1] = 5000000
extreme_large: (-2, .., -2, 1, .., 1) and (MAX_INT)...(MAX_INT), length ~100000
"""
def test_example(self):
self.assertEqual(solution([-3, 1, 2, -2, 5, 6]), 60)
def test_sample(self):
self.assertEqual(solution([-5, 5, -5, 4]), 125)
def test_zero(self):
self.assertEqual(solution([0, 0, 0, 0]), 0)
self.assertEqual(solution([-10, -10, -3, 0, 1]), 100)
def test_negative(self):
self.assertEqual(solution([-4, -3, -2, -1]), -6)
self.assertEqual(solution([-1, -1, 1, 2]), 2)
self.assertEqual(solution([-5, -5, 1, 2]), 50)
self.assertEqual(solution([-5, -5, -1, 0]), 0)
def test_large(self):
self.assertEqual(solution([1000, 1000, 1000]), int(1e9))
def test_extreme(self):
A = [random.randint(*RANGE_A) for _ in range(3, 99997)]
A += [1000, 1000, 1000]
self.assertEqual(solution(A), int(1e9))
if __name__ == '__main__':
unittest.main()