NoobCoder1998's blog

By NoobCoder1998, history, 6 hours ago, In English

You are given 2 arrays. Choose a subarray from the first array (arr1) and replace it with the corresponding subarray from the second array (arr2). After replacing the subarray, calculate the maximum consecutive subarray sum of the modified first array. Find the maximum subarray sum you can get by doing this operation. You can do this operation only once.

N is 10^5 and ai [-1e9, 1e9]

Im not able to think of better than n^2. Pls help

  • Vote: I like it
  • +3
  • Vote: I do not like it

»
6 hours ago, # |
  Vote: I like it 0 Vote: I do not like it

Auto comment: topic has been updated by NoobCoder1998 (previous revision, new revision, compare).

»
6 hours ago, # |
  Vote: I like it 0 Vote: I do not like it

Kadane along with sliding window should do the trick in O(N)

Ps: Can you provide problem name or number.

  • »
    »
    6 hours ago, # ^ |
      Vote: I like it 0 Vote: I do not like it

    This was asked to me in Online assessment so cant provide link.. can u share ur approach

»
5 hours ago, # |
Rev. 2   Vote: I like it +6 Vote: I do not like it

The best solution I can think of:

You can solve it using DP. Let $$$dp[i][x]$$$ be the maximum sum of a subarray starting at index $$$i$$$, where:

  • $$$x=0$$$ means the replaced subarray begins after the current index.
  • $$$x=1$$$ means this index is part of the replaced subarray.
  • $$$x=2$$$ means the replaced subarray ends before the current index.

If $$$x=0$$$, we can:

  • Decide to start the replaced subarray on the current element ($$$dp[i][1]$$$),
  • Add the current value (arr1[i]) to the sum and go to the next element ($$$dp[i+1][0]+arr1[i]$$$), or
  • Decide not to add any more elements to the final subarray (the one with the maximum sum) ($$$0$$$).

If $$$x=1$$$, we can:

  • Decide to end the replaced subarray on the previous element (so this will be the first element that won't be a part of the subarray) ($$$dp[i][2]$$$),
  • Add the current value (arr2[i]) to the sum and go to the next element ($$$dp[i+1][1]+arr2[i]$$$), or
  • Decide not to add any more elements to the final subarray (the one with the maximum sum) ($$$0$$$).

If $$$x=2$$$, we can:

  • Add the current value (arr1[i]) to the sum and go to the next element ($$$dp[i+1][2]+arr1[i]$$$), or
  • Decide not to add any more elements to the final subarray (the one with the maximum sum) ($$$0$$$).

So:
$$$dp[i][0]=max(dp[i][1],dp[i+1][0]+arr1[i],0)$$$
$$$dp[i][1]=max(dp[i][2],dp[i+1][1]+arr2[i],0)$$$
$$$dp[i][2]=max(dp[i+1][2]+arr1[i],0)$$$
Base case: $$$dp[n][x]=0$$$

The answer is the maximum value of $$$dp[i][0]$$$ for all $$$i$$$.

Total complexity: $$$O(n)$$$

Code

Hope it helps!

  • »
    »
    4 hours ago, # ^ |
      Vote: I like it 0 Vote: I do not like it

    Thanks for the solution.. after some thinking i came up with one approach

    Replacing a subarray in arr1 with a corresponding subarray from arr2 is equivalent to calculating the difference between the subarrays in arr2 and arr1. For example, let arr1 = [1,2,3], arr2 = [1,4,5]. If I want to replace subarray[2,3] with [4,5] that will change the subarray sum from 5 to 9, increase by 4. by calculating difference of values in the subarray also you can get this. (4-2)+(5-3) = 4. So, you can calculate the entire difference array and use kadane's algo to come at a maximum sum subarray. This subarray will denote the start and end in the original arrays that needs to be replaced.

    Will this fail ??

    • »
      »
      »
      3 hours ago, # ^ |
        Vote: I like it 0 Vote: I do not like it

      I thought of calculating a difference array, too, at first, but wasn't sure how to use it afterwards. I had forgotten about Kadane's algorithm. I just looked it up and, although I understand the algorithm itself, I didn't fully understand how to use it in this specific problem. Can you explain how it's used in your approach?

      • »
        »
        »
        »
        3 hours ago, # ^ |
          Vote: I like it 0 Vote: I do not like it

        So my approach is to create a difference array first like if a1= [1,4,2,6,8] a2= [6,4,8,1,2] On calculating difference array we get Diff = [5,0,6,-5,-6] Now if we run kadanes on this we get subarray (0,2) which is 11. So we conclude we need to replace (0,2) Subarray so a1 becomes [6,4,8,6,8]

        • »
          »
          »
          »
          »
          116 minutes ago, # ^ |
            Vote: I like it 0 Vote: I do not like it

          This doesn't work if, for example, a1=[-2, -2, -2, -100, 4, 5] and a2=[-1, -1, -1, -10000, 5, 6].

          The difference array is [1, 1, 1, -9900, 1, 1], so the optimal subarray to replace is (0,2). The new a1 will be [-1, -1, -1, -100, 4, 5], making the maximum subarray sum 9 by choosing subarray (4,5).

          However, by replacing subarray (4,5) we get a1=[-2, -2, -2, -100, 5, 6] with maximum sum 11 by choosing the subarray (4,5).

          So it's not enough to choose the subarray of the difference array with the maximum sum.

»
5 hours ago, # |
Rev. 2   Vote: I like it 0 Vote: I do not like it

I am not sure whether this is correct or not, but I think you can use dp and Kadane algorithm

the dp states are i and j where i is the index and j is $$$0$$$ if you haven't taken any element from $$$b$$$, $$$1$$$ if the previous element was from $$$b$$$ (i.e. you can still take element $$$i$$$ from $$$b$$$) and $$$2$$$ if you can no longer take elements from $$$b$$$

$$$dp_{sum}[i][j]$$$ is the sum until state $$$i, j$$$, $$$dp_{max}[i][j]$$$ is the maximum subarray sum till state $$$i, j$$$.

here is my implementation to this:

void solve() {
    int n;
    cin >> n;
    vector<int> a(n + 1), b(n + 1);
    for (int i = 1; i <= n; ++i) cin >> a[i];
    for (int i = 1; i <= n; ++i) cin >> b[i];
    const int canTake = 0, taking = 1, cant = 2;
    vector dpSum(n + 1, vector<ll>(3));
    vector dpMax(n + 1, vector<ll>(3));
    for (int i = 1; i <= n; ++i) {
        for (int j = 0; j < 3; ++j) {
            if (j == canTake){
                dpSum[i][canTake] = dpSum[i - 1][canTake];
                if (a[i] > dpSum[i][canTake]){
                    dpSum[i][canTake] = a[i];
                } else {
                    dpSum[i][canTake] += a[i];
                }
                dpMax[i][canTake] = max(dpMax[i - 1][canTake], dpSum[i][canTake]);
            } else if (j == taking){
                dpSum[i][taking] = max(dpSum[i - 1][canTake], dpSum[i - 1][taking]);
                if (b[i] > dpSum[i][taking]){
                    dpSum[i][taking] = b[i];
                } else {
                    dpSum[i][taking] += b[i];
                }
                dpMax[i][taking] = max({dpMax[i - 1][canTake], dpMax[i - 1][taking], dpSum[i][taking]});
            } else {
                dpSum[i][cant] = max(dpSum[i - 1][cant], dpSum[i - 1][taking]);
                if (a[i] > dpSum[i][cant]){
                    dpSum[i][cant] = a[i];
                } else {
                    dpSum[i][cant] += a[i];
                }
                dpMax[i][cant] = max({dpMax[i - 1][cant], dpMax[i - 1][taking], dpSum[i][cant]});
            }
        }
    }
    cout << max({dpMax[n][0], dpMax[n][1], dpMax[n][2]}) << endl;
}

I tried it on some randomly generated tests and it works