-
Notifications
You must be signed in to change notification settings - Fork 154
/
Copy pathk-workers.js
117 lines (102 loc) · 3.8 KB
/
k-workers.js
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
/**
* K Workers
*
* There are n workers, each worker can finish the work with certain quality, and each worker asks for different price.
* Given these n workers, you are asked to select k (0 < k <= n) workers, and make sure the total price is minimal.
*
* However, there is a rule, whenever you add a new worker, if that worker's price/quality ratio is higher than workers that
* have been selected, those selected workers won't be happy and they will ask you for higher price, and the way to calculate
* the new price is: new price = original quality x new worker's price/quality ratio.
*
* For example, let's say you have selected worker A, A's price is $5, his quality is 20, now you added worker B, B's price is
* $10, and his quality is 10, since B's price/quality ratio is higher than A's price/quality ratio, A is asking for new
* price, and this time, A will ask for 20 x $10/10 = $20, now hiring both A and B will cost you $20 + $10 = $30.
*
* The input of the problem are: k, workers[[price, quality]], and the output of the problem should be the minimal total price.
*
* Example 1:
* workers = [[10, 20], [20, 10], [5, 10], [4, 8]]
* k = 2
*
* You should return 9
* Explanation: the two workers you should select is [4, 8], [5, 10], their total price is 9
*
* Example 2:
* workers = [[4, 16], [5, 10], [1, 1]]
* k = 2
* You should return 11
* Explanation: the two workers you should select is [5, 10], [1, 1]
*
* Example 3:
* workers = [[4, 16], [5, 10], [6, 6]]
* k = 2
* You should return 13
* Explanation: the two workers you should select is [4, 16], [5, 10]
*/
import PriorityQueue from 'common/priority-queue';
/**
* @param {number} k
* @param {number[]} workers
* @return {number}
*/
const kWorkers = (k, workers) => {
if (k <= 0 || !workers || k > workers.length) {
return 0;
}
// Step 1. sort workers by their price/quality ratio
// Time complexity: nlogn
sortWorkers(workers);
// Step 2. start from the kth worker, calculate the current total cost and see if it's the minimal
// in the meantime, if the kth worker's quality is less than the max quality among the k - 1 workers
// replace the two workers and continue
// Time complexity: (n - k) * (k + logk)
let minPrice = Number.MAX_SAFE_INTEGER;
// Initialize the max heap for helping us know the worker with max quality
const queue = new PriorityQueue({ comparator: (a, b) => b.worker[1] - a.worker[1] });
for (let i = 0; i < k - 1; i++) {
queue.offer({ index: i, worker: workers[i] });
}
for (let i = k - 1; i < workers.length; i++) {
const kthWorker = workers[i];
const kthRatio = kthWorker[0] / kthWorker[1]; // price/quality
// Sum up the k - 1 workers' prices with the new ratio
let currentPrice = 0;
for (let j = 0; j < k - 1; j++) {
currentPrice += workers[j][1] * kthRatio;
}
// Plus the k-th worker
currentPrice += kthWorker[0];
// Calculate the minimum price
minPrice = Math.min(minPrice, currentPrice);
// if k-th worker's quality is less, replace the two
if (queue.size() === 0) {
queue.offer({ index: i, worker: kthWorker });
} else {
// Get the worker with the max quality
const { index, worker } = queue.peek();
if (worker[1] > kthWorker[1]) {
queue.poll();
queue.offer({ index: i, worker: kthWorker });
// Replace the workers in the array
swap(workers, index, i);
}
}
}
return minPrice;
};
/**
* Sort the workers by their price/quality ratio
* @param {number[]} workers
*/
const sortWorkers = workers => {
workers.sort((a, b) => {
const aRatio = a[0] / a[1];
const bRatio = b[0] / b[1];
if (aRatio === bRatio) {
return a[0] - b[0];
}
return aRatio - bRatio;
});
};
const swap = (arr, i, j) => ([arr[i], arr[j]] = [arr[j], arr[i]]);
export { kWorkers };