[codeforces1219G]Harvester

time limit per test : 0.5 seconds
memory limit per test : 256 megabytes

It is Bubble Cup finals season and farmer Johnny Bubbles must harvest his bubbles. The bubbles are in a rectangular bubblefield formed of NNxMM square parcels divided into NN rows and MM columns. The parcel in ith row and jth column yields Ai,jA_{i,j} bubbles.

Johnny Bubbles has available a very special self-driving bubble harvester that, once manually positioned at the beginning of a row or column, automatically harvests all the bubbles in that row or column. Once the harvester reaches the end of the row or column it stops and must be repositioned. The harvester can pass through any parcel any number of times, but it can collect bubbles from the parcel only once.

Johnny is very busy farmer, so he is available to manually position the harvester at most four times per day. Johnny is also impatient, so he wants to harvest as many bubbles as possible on the first day.
Please help Johnny to calculate what is the maximum number of bubbles he can collect on the first day.

Input

The first line contains two integers NN and M(1N,MNM105)M (1 ≤ N, M ≤ N * M ≤ 10^5) - the bubblefield size.

Each of the next NN lines contains M integers. The jth element in the ith line is Ai,j(0Ai,j109)A_{i,j} (0 ≤ A_{i,j} ≤ 10^9) — the yield of the parcel located in the ith row and the jj-th column.

Output

Output contains one integer number - maximum number of the bubbles Johnny can harvest on the first day.
Examples
Input

2 2
1 2
3 4

Output

10

Input

5 5
0 9 2 7 0
9 0 3 0 5
0 8 0 3 1
6 7 4 3 9
3 6 4 1 0

Output

80

Note

In the first example, farmer Johnny can harvest all the bubbles by positioning the harvester on the first and the second row.

In the second example, one way Johnny can harvest maximum number of bubbles is to position the harvester in the second row, the fourth row, the second column and the fourth column.

題意:
給定一個NM<=105N*M<=10^5的矩陣,詢問從矩陣上最多取出行列一共4個,問取出元素的最大值是多少。

題解:
分5種情況:
1:全選行,直接做
2:全選列,直接做
3:一行三列,枚舉行,維護去掉行的最大的三個行的值O(NM)O(N*M)
4:一列三行,枚舉列,維護去掉列的最大的三個行的值O(NM)O(N*M)
5:兩行兩列,枚舉行列中短的那個,然後掃一遍維護另一個的最大次大值。O(min(N,M)2max(N,M))O(min(N,M)^2*max(N,M))

#include<bits/stdc++.h>
#define ll long long
using namespace std;
int n,m;
inline int pos(int x,int y){return (x-1)*m+y;}
int a[100004],a2[100004];
ll scol[100004],srow[100004];
priority_queue<ll,vector<ll>,greater<ll> >q;
ll getcol(){
    while(!q.empty())q.pop();
    for(int i=1;i<=m;i++){
        ll now=0;
        if(q.size()<4){
            q.push(scol[i]);
        }
        else if(q.top()<scol[i]){
            q.pop();
            q.push(scol[i]);
        }
    }
    ll ret=0;
    while(!q.empty()){
        ret+=q.top();q.pop();
    }
    return ret;
}
ll getrow(){
    while(!q.empty())q.pop();
    for(int i=1;i<=n;i++){
        ll now=0;
        if(q.size()<4){
            q.push(srow[i]);
        }
        else if(q.top()<srow[i]){
            q.pop();
            q.push(srow[i]);
        }
    }
    ll ret=0;
    while(!q.empty()){
        ret+=q.top();q.pop();
    }
    return ret;
}
ll sekcol(){

    ll ret=0;
    for(int i=1;i<=m;i++){
        ll now=scol[i];
        while(!q.empty())q.pop();
        for(int j=1;j<=n;j++){
            if(q.size()<3){
                q.push(srow[j]-a[pos(j,i)]);
            }
            else if(q.top()<srow[j]-a[pos(j,i)]){
                q.pop();
                q.push(srow[j]-a[pos(j,i)]);
            }
        }
        while(!q.empty()){
            now+=q.top();q.pop();
        }
        ret=max(now,ret);
    }
    return ret;
}
ll sekrow(){

    ll ret=0;
    for(int i=1;i<=n;i++){
        ll now=srow[i];
        while(!q.empty())q.pop();
        for(int j=1;j<=n;j++){
            if(q.size()<3){
                q.push(scol[j]-a[pos(i,j)]);
            }
            else if(q.top()<scol[j]-a[pos(i,j)]){
                q.pop();
                q.push(scol[j]-a[pos(i,j)]);
            }
        }
        while(!q.empty()){
            now+=q.top();q.pop();
        }
        ret=max(now,ret);
    }
    return ret;
}
ll work(){
    ll ret=0;
    for(int i=1;i<=n;i++){
        for(int j=i+1;j<=n;j++){
            ll now=srow[i]+srow[j];
            while(!q.empty())q.pop();
            for(int k=1;k<=m;k++){
                if(q.size()<2){
                    q.push(scol[k]-a[pos(i,k)]-a[pos(j,k)]);
                }
                else if(q.top()<scol[k]-a[pos(i,k)]-a[pos(j,k)]){
                    q.pop();
                    q.push(scol[k]-a[pos(i,k)]-a[pos(j,k)]);
                }
            }
            while(!q.empty()){
                now+=q.top();q.pop();
            }
            ret=max(now,ret);
        }
    }
    return ret;
}
int main(){
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++)
        for(int j=1;j<=m;j++)
            scanf("%d",&a2[pos(i,j)]);
    if(n>m){
        for(int i=1;i<=n;i++){
            for(int j=1;j<=m;j++){
                a[(j-1)*n+i]=a2[pos(i,j)];
            }
        }
        swap(n,m);
    }
    else{
        for(int i=1;i<=n;i++){
            for(int j=1;j<=m;j++){
                a[pos(i,j)]=a2[pos(i,j)];
            }
        }
    }
    for(int i=1;i<=n;i++){
        for(int j=1;j<=m;j++){
            int v=a[pos(i,j)];
            srow[i]+=v;
            scol[j]+=v;
        }
    }
    ll ans=0;
    ans=max(ans,getcol());
    ans=max(ans,getrow());
    ans=max(ans,sekcol());
    ans=max(ans,sekrow());
    ans=max(ans,work());
    printf("%lld\n",ans);
    return 0;
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章