ÐодмаÑÑив наиболÑÑей ÑÑммÑ
Ðа вÑ
оде маÑÑив ÑиÑел, напÑимеÑ: arr = [1, -2, 3, 4, -9, 6].
ÐадаÑа: найÑи непÑеÑÑвнÑй подмаÑÑив в arr, ÑÑмма ÑлеменÑов в коÑоÑом макÑималÑна.
ФÑнкÑÐ¸Ñ getMaxSubSum(arr) должна возвÑаÑаÑÑ ÑÑÑ ÑÑммÑ.
ÐапÑимеÑ:
getMaxSubSum([-1, 2, 3, -9]) == 5 (ÑÑмма вÑделеннÑÑ
ÑлеменÑов)
getMaxSubSum([2, -1, 2, 3, -9]) == 6
getMaxSubSum([-1, 2, 3, -9, 11]) == 11
getMaxSubSum([-2, -1, 1, 2]) == 3
getMaxSubSum([100, -9, 2, -3, 5]) == 100
getMaxSubSum([1, 2, 3]) == 6 (беÑÑм вÑе)
ÐÑли вÑе ÑлеменÑÑ Ð¾ÑÑиÑаÑелÑнÑе â ниÑего не беÑÑм(подмаÑÑив пÑÑÑой) и ÑÑмма Ñавна «0»:
getMaxSubSum([-1, -2, -3]) = 0
ÐопÑобÑйÑе пÑидÑмаÑÑ Ð±ÑÑÑÑое ÑеÑение: O(n2), а лÑÑÑе за Ð(n) опеÑаÑий.
ÐÑкÑÑÑÑ Ð¿ÐµÑоÑниÑÑ Ñ ÑеÑÑами Ð´Ð»Ñ Ð·Ð°Ð´Ð°Ñи.
Ðедленное ÑеÑение
Ðожно поÑÑиÑаÑÑ Ð²Ñе возможнÑе подÑÑммÑ.
СамÑй пÑоÑÑой пÑÑÑ â поÑÑиÑаÑÑ ÑÑÐ¼Ð¼Ñ Ð¿Ð¾Ð´Ð¼Ð°ÑÑивов, наÑÐ¸Ð½Ð°Ñ Ñ ÐºÐ°Ð¶Ð´Ð¾Ð³Ð¾ ÑлеменÑа по оÑеÑеди.
ÐапÑимеÑ, Ð´Ð»Ñ [-1, 2, 3, -9, 11]:
// ÐаÑÐ¸Ð½Ð°Ñ Ñ -1:
-1
-1 + 2
-1 + 2 + 3
-1 + 2 + 3 + (-9)
-1 + 2 + 3 + (-9) + 11
// ÐаÑÐ¸Ð½Ð°Ñ Ñ 2:
2
2 + 3
2 + 3 + (-9)
2 + 3 + (-9) + 11
// ÐаÑÐ¸Ð½Ð°Ñ Ñ 3:
3
3 + (-9)
3 + (-9) + 11
// ÐаÑÐ¸Ð½Ð°Ñ Ñ -9
-9
-9 + 11
// ÐаÑÐ¸Ð½Ð°Ñ Ñ 11
11
РеализÑеÑÑÑ Ñ Ð¿Ð¾Ð¼Ð¾ÑÑÑ Ð²Ð»Ð¾Ð¶ÐµÐ½Ð½Ð¾Ð³Ð¾ Ñикла: внеÑний Ñикл пÑÐ¾Ñ Ð¾Ð´Ð¸Ñ Ð¿Ð¾ ÑлеменÑам маÑÑива, а внÑÑÑенний ÑÑиÑÐ°ÐµÑ Ð¿Ð¾Ð´ÑÑммÑ, наÑÐ¸Ð½Ð°Ñ Ñ ÑекÑÑего ÑлеменÑа.
function getMaxSubSum(arr) {
let maxSum = 0; // еÑли ÑлеменÑов не бÑÐ´ÐµÑ - возвÑаÑаем 0
for (let i = 0; i < arr.length; i++) {
let sumFixedStart = 0;
for (let j = i; j < arr.length; j++) {
sumFixedStart += arr[j];
maxSum = Math.max(maxSum, sumFixedStart);
}
}
return maxSum;
}
alert( getMaxSubSum([-1, 2, 3, -9]) ); // 5
alert( getMaxSubSum([-1, 2, 3, -9, 11]) ); // 11
alert( getMaxSubSum([-2, -1, 1, 2]) ); // 3
alert( getMaxSubSum([1, 2, 3]) ); // 6
alert( getMaxSubSum([100, -9, 2, -3, 5]) ); // 100
ÐÑо ÑеÑение Ð¸Ð¼ÐµÐµÑ Ð¾ÑÐµÐ½ÐºÑ ÑложноÑÑи O(n2). ÐÑÑгими Ñловами, еÑли Ð¼Ñ ÑвелиÑим ÑÐ°Ð·Ð¼ÐµÑ Ð¼Ð°ÑÑива в 2 Ñаза, вÑÐµÐ¼Ñ Ð²ÑÐ¿Ð¾Ð»Ð½ÐµÐ½Ð¸Ñ Ð°Ð»Ð³Ð¾ÑиÑма ÑвелиÑиÑÑÑ Ð² 4 Ñаза.
ÐÐ»Ñ Ð±Ð¾Ð»ÑÑÐ¸Ñ Ð¼Ð°ÑÑивов(1000, 10000 или болÑÑе ÑлеменÑов) Ñакие алгоÑиÑÐ¼Ñ Ð¼Ð¾Ð³ÑÑ Ð¿ÑиводиÑÑ Ðº ÑеÑÑÑзнÑм «ÑоÑмозам».
ÐÑÑÑÑое ÑеÑение
ÐдÑм по маÑÑÐ¸Ð²Ñ Ð¸ накапливаем ÑекÑÑÑÑ ÑаÑÑиÑнÑÑ ÑÑÐ¼Ð¼Ñ ÑлеменÑов в пеÑеменной s. ÐÑли s в какой-Ñо Ð¼Ð¾Ð¼ÐµÐ½Ñ ÑÑановиÑÑÑ Ð¾ÑÑиÑаÑелÑной â пÑиÑваиваем s=0. ÐакÑималÑнÑй из вÑеÑ
s и бÑÐ´ÐµÑ Ð¾ÑвеÑом.
ÐÑли обÑÑÑнение недоÑÑаÑоÑно понÑÑно, поÑмоÑÑиÑе на код, он вполне лакониÑен:
function getMaxSubSum(arr) {
let maxSum = 0;
let partialSum = 0;
for (let item of arr) { // Ð´Ð»Ñ ÐºÐ°Ð¶Ð´Ð¾Ð³Ð¾ ÑлеменÑа маÑÑива
partialSum += item; // добавлÑем знаÑение ÑлеменÑа к partialSum
maxSum = Math.max(maxSum, partialSum); // запоминаем макÑимÑм на даннÑй моменÑ
if (partialSum < 0) partialSum = 0; // Ð½Ð¾Ð»Ñ ÐµÑли оÑÑиÑаÑелÑное
}
return maxSum;
}
alert( getMaxSubSum([-1, 2, 3, -9]) ); // 5
alert( getMaxSubSum([-1, 2, 3, -9, 11]) ); // 11
alert( getMaxSubSum([-2, -1, 1, 2]) ); // 3
alert( getMaxSubSum([100, -9, 2, -3, 5]) ); // 100
alert( getMaxSubSum([1, 2, 3]) ); // 6
alert( getMaxSubSum([-1, -2, -3]) ); // 0
ÐÑÐ¾Ñ Ð°Ð»Ð³Ð¾ÑиÑм ÑÑебÑÐµÑ Ñовно 1 пÑÐ¾Ñ Ð¾Ð´ по маÑÑÐ¸Ð²Ñ Ð¸ его оÑенка ÑложноÑÑи O(n).
ÐолÑÑе инÑоÑмаÑии об алгоÑиÑме ÑÑÑ: ÐадаÑа поиÑка макÑималÑной ÑÑÐ¼Ð¼Ñ Ð¿Ð¾Ð´Ð¼Ð°ÑÑива. ÐÑли вÑÑ ÐµÑÑ Ð½Ðµ оÑевидно как ÑÑо ÑабоÑаеÑ, пÑоÑмоÑÑиÑе алгоÑиÑм в пÑимеÑÐ°Ñ Ð²ÑÑе, ÑÑо бÑÐ´ÐµÑ Ð»ÑÑÑе вÑÑÐºÐ¸Ñ Ñлов.
ÐÑкÑÑÑÑ ÑеÑение Ñ ÑеÑÑами в пеÑоÑниÑе.