For a given array A of N integers and a sequence S of N integers from the set {−1, 1}, we define val(A, S) as follows:
val(A, S) = |sum{ A[i]*S[i] for i = 0..N−1 }|
(Assume that the sum of zero elements equals zero.)
For a given array A, we are looking for such a sequence S that minimizes val(A,S).
Write a function:
function solution(A);
that, given an array A of N integers, computes the minimum value of val(A,S) from all possible values of val(A,S) for all possible sequences S of N integers from the set {−1, 1}.
For example, given array:
A[0] = 1 A[1] = 5 A[2] = 2 A[3] = -2your function should return 0, since for S = [−1, 1, −1, 1], val(A, S) = 0, which is the minimum possible value.
Write an efficient algorithm for the following assumptions:
- N is an integer within the range [0..20,000];
- each element of array A is an integer within the range [−100..100].
// you can write to stdout for debugging purposes, e.g.
// console.log('this is a debug message');
function solution(A) {
A.forEach((value, index, array) => {
array[index] = Math.abs(value)
})
const maxValue = Math.max(...A)
const sameValuesCount = new Array(maxValue + 1)
sameValuesCount.fill(0)
A.forEach((value) => {
++sameValuesCount[value]
})
const maxSum = sameValuesCount.reduce((previousValue, currentValue, currentIndex) => {
return previousValue + currentValue * currentIndex
}, 0)
const dp = new Array(maxSum + 1)
dp.fill(-1)
dp[0] = 0
sameValuesCount.forEach((countValue, countIndex) => {
if (countValue > 0) {
dp.forEach((dpValue, dpIndex, dpArray) => {
if (dpValue >= 0) {
dpArray[dpIndex] = countValue
} else if (dpIndex >= countIndex && dpArray[dpIndex - countIndex] > 0) {
dpArray[dpIndex] = dpArray[dpIndex - countIndex] - 1
}
})
}
})
let result = maxSum
for (let i = 0; i < Math.floor(maxSum / 2) + 1; ++i) {
if (dp[i] >= 0) {
result = Math.min(result, maxSum - 2 * i)
}
}
return result
}
// you can write to stdout for debugging purposes, e.g.
// console.log('this is a debug message');
function solution(A) {
A.forEach((value, index, array) => {
array[index] = Math.abs(value)
})
const maxValue = Math.max(...A)
const sameValuesCount = new Array(maxValue + 1)
sameValuesCount.fill(0)
A.forEach((value) => {
++sameValuesCount[value]
})
const maxSum = sameValuesCount.reduce((previousValue, currentValue, currentIndex) => {
return previousValue + currentValue * currentIndex
}, 0)
const dp = new Array(maxSum + 1)
dp.fill(-1)
dp[0] = 0
sameValuesCount.forEach((countValue, countIndex) => {
if (countValue > 0) {
dp.forEach((dpValue, dpIndex, dpArray) => {
if (dpValue >= 0) {
dpArray[dpIndex] = countValue
} else if (dpIndex >= countIndex && dpArray[dpIndex - countIndex] > 0) {
dpArray[dpIndex] = dpArray[dpIndex - countIndex] - 1
}
})
}
})
let minSum = maxSum
for (let i = Math.floor(maxSum / 2); i >= 0; --i) {
if (dp[i] >= 0) {
minSum = Math.min(minSum, maxSum - 2 * i)
break
}
}
return minSum
}
// you can write to stdout for debugging purposes, e.g.
// console.log('this is a debug message');
function solution(A) {
A.forEach((value, index, array) => {
array[index] = Math.abs(value)
})
const maxValue = Math.max(...A)
const sameValuesCount = new Array(maxValue + 1)
sameValuesCount.fill(0)
A.forEach((value) => {
++sameValuesCount[value]
})
const maxSum = sameValuesCount.reduce((previousValue, currentValue, currentIndex) => {
return previousValue + currentValue * currentIndex
}, 0)
const dp = new Array(maxSum + 1)
dp.fill(-1)
dp[0] = 0
sameValuesCount.forEach((countValue, countIndex) => {
if (countValue > 0) {
dp.forEach((dpValue, dpIndex, dpArray) => {
if (dpValue >= 0) {
dpArray[dpIndex] = countValue
} else if (dpIndex >= countIndex && dpArray[dpIndex - countIndex] > 0) {
dpArray[dpIndex] = dpArray[dpIndex - countIndex] - 1
}
})
}
})
let minSum = maxSum
for (let i = Math.floor(maxSum / 2); i >= 0; --i) {
if (dp[i] >= 0) {
minSum = Math.min(minSum, maxSum - 2 * i)
break
}
}
return minSum
}
// you can write to stdout for debugging purposes, e.g.
// console.log('this is a debug message');
function solution(A) {
A.forEach((value, index, array) => {
array[index] = Math.abs(value)
})
const maxValue = Math.max(...A)
const sameValuesCount = new Array(maxValue + 1)
sameValuesCount.fill(0)
A.forEach((value) => {
++sameValuesCount[value]
})
const maxSum = sameValuesCount.reduce((previousValue, currentValue, currentIndex) => {
return previousValue + currentValue * currentIndex
}, 0)
const dp = new Array(maxSum + 1)
dp.fill(-1)
dp[0] = 0
sameValuesCount.forEach((countValue, countIndex) => {
if (countValue > 0) {
dp.forEach((dpValue, dpIndex, dpArray) => {
if (dpValue >= 0) {
dpArray[dpIndex] = countValue
} else if (dpIndex >= countIndex && dpArray[dpIndex - countIndex] > 0) {
dpArray[dpIndex] = dpArray[dpIndex - countIndex] - 1
}
})
}
})
let minSum = maxSum
for (let i = Math.floor(maxSum / 2); i >= 0; --i) {
if (dp[i] >= 0) {
minSum = Math.min(minSum, maxSum - 2 * i)
break
}
}
return minSum
}
// you can write to stdout for debugging purposes, e.g.
// console.log('this is a debug message');
function solution(A) {
A.forEach((value, index, array) => {
array[index] = Math.abs(value)
})
const maxValue = Math.max(...A)
const sameValuesCount = new Array(maxValue + 1)
sameValuesCount.fill(0)
A.forEach((value) => {
++sameValuesCount[value]
})
const maxSum = sameValuesCount.reduce((previousValue, currentValue, currentIndex) => {
return previousValue + currentValue * currentIndex
}, 0)
const dp = new Array(maxSum + 1)
dp.fill(-1)
dp[0] = 0
sameValuesCount.forEach((countValue, countIndex) => {
if (countValue > 0) {
dp.forEach((dpValue, dpIndex, dpArray) => {
if (dpValue >= 0) {
dpArray[dpIndex] = countValue
} else if (dpIndex >= countIndex && dpArray[dpIndex - countIndex] > 0) {
dpArray[dpIndex] = dpArray[dpIndex - countIndex] - 1
}
})
}
})
let minSum = maxSum
for (let i = Math.floor(maxSum / 2); i >= 0; --i) {
if (dp[i] >= 0) {
minSum = Math.min(minSum, maxSum - 2 * i)
break
}
}
return minSum
}
The following issues have been detected: runtime errors.
For example, for the input [] the solution terminated unexpectedly.
RangeError: Invalid array length at solution (solution.js:12:29) at solutionWrapper (/tmp/exec.js:402:28) at Promise.resolve.then (/tmp/exec.js:428:24) at <anonymous> at process._tickCallback (internal/process/next_tick.js:188:7) at Function.Module.runMain (module.js:686:11) at startup (bootstrap_node.js:187:16) at bootstrap_node.js:608:3