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 x square parcels divided into rows and columns. The parcel in ith row and jth column yields 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 and - the bubblefield size.
Each of the next lines contains M integers. The jth element in the ith line is — the yield of the parcel located in the ith row and the -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.
題意:
給定一個的矩陣,詢問從矩陣上最多取出行列一共4個,問取出元素的最大值是多少。
題解:
分5種情況:
1:全選行,直接做
2:全選列,直接做
3:一行三列,枚舉行,維護去掉行的最大的三個行的值
4:一列三行,枚舉列,維護去掉列的最大的三個行的值
5:兩行兩列,枚舉行列中短的那個,然後掃一遍維護另一個的最大次大值。
#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;
}