码迷,mamicode.com
首页 > 其他好文 > 详细

JZOJ.5307【NOIP2017模拟8.18】偷窃

时间:2017-08-18 19:54:19      阅读:192      评论:0      收藏:0      [点我收藏+]

标签:heap   wcry   bean   unp   avr   jni   psi   vdp   psk   

Description

技术分享
 

Input

技术分享

Output

技术分享
 

Sample Input

5 5
1 4 0 5 2
2 1 2 0 1
0 2 3 4 4
0 3 0 3 1
1 2 2 1 1

Sample Output

9
 

Data Constraint

技术分享
 

Hint

技术分享

本题直接做比较麻烦,加上金砖可移动。

这题就是要求我们找出一种摆放方式,三视图与之前一样且用得金块数最少。

我们可以重新构造。

对于俯视图,它提供的信息就是我们要在哪里放金砖,哪里不能放金砖。

对于正视图和侧视图,它提供的信息就是某一行或某一列的最大高度有多高。

那么我们就根据这三视图提供的信息一一构造。

对于俯视图,我们就将一开始为0的地方就视为不可放东西。

要求放置的金砖数最少,自然我们想某一列的最大高度与某一行的最大高度一样,这样我们就用那么高的金砖就能满足正视图和侧视图的某一行列的高度。

很显然这是最好的方案。

于是我们统计每一行的最大高度和每一列的最大高度,看看能不能把某一行和某一列的同样的高度用一个高度的金砖”柱子“来代替。

这就是二分图匹配啦~

行在左,列在右,相等高度一连线,跑个匈牙利算法,能匹配的高度一次算,未能匹配后加上,再把剩下的非零的格子都视为1,统计下金砖数和一开始的总金砖数做个差就是答案了。

技术分享
 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #define N 105
 5 using namespace std;
 6 int f[N][N],maxl[N],maxh[N],f1[N],f2[N],used[N];
 7 long long ans,sum,n,m,zero,ji;
 8 bool find(int x){
 9     for (int i=1;i<=m;i++)
10      if ((maxl[i]==maxh[x])&&(used[i]==false))
11       if ((maxl[i]==0)||(f[x][i]!=0)){
12          used[i]=true;
13          if ((f1[i]==0)||find(f1[i])){
14              f1[i]=x;
15              f2[x]=i;
16              return true;
17          }
18      }
19      return false;
20 }
21 int main(){
22     scanf("%lld%lld",&n,&m);
23     for (int i=1;i<=n;i++)
24      for (int j=1;j<=m;j++){
25          scanf("%d",&f[i][j]);
26          sum+=f[i][j];
27          if (f[i][j]==0) zero++;
28          maxl[j]=max(maxl[j],f[i][j]);
29          maxh[i]=max(maxh[i],f[i][j]);
30      }
31     ans=0;
32     for (int i=1;i<=n;i++){
33         memset(used,false,sizeof(used));
34         if (find(i)) {ans+=maxh[i];
35         if (f[i][f2[i]]!=0) ji++;
36         }
37     }
38     for (int i=1;i<=n;i++)
39         if ((f2[i]==0)&&(maxh[i]!=0)) ans+=maxh[i],ji++;
40     for (int i=1;i<=m;i++)
41         if ((f1[i]==0)&&(maxl[i]!=0)) ans+=maxl[i],ji++;
42     printf("%lld\n",sum-ans-(n*m-zero-ji));
43     return 0;
44 }
神奇的代码

其实这题还能贪心,把每行每列最高的金砖地方做个标记,然后再看看能不能删去一些金砖即可。

写的好的能骗个100分。

然而我这个当时只能骗到个80分qwq

技术分享
 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #define N 102
 5 using namespace std;
 6 int n,m,hang[N],lie[N],maxh[N],maxl[N],sigh[N][N],f[N][N];
 7 int main(){
 8     scanf("%d%d",&n,&m);
 9     long long tmp=0;
10     for (int i=1;i<=n;i++)
11      for (int j=1;j<=m;j++){
12          scanf("%d",&f[i][j]);
13          maxl[j]=max(maxl[j],f[i][j]);
14          maxh[i]=max(maxh[i],f[i][j]);
15          tmp+=f[i][j];
16      }
17      if (tmp==0) {printf("0\n");return 0;}
18      for (int i=1;i<=n;i++)
19       for (int j=1;j<=m;j++){
20           if (f[i][j]==maxh[i])
21               hang[i]++,sigh[i][j]++;
22           if (f[i][j]==maxl[j])
23               sigh[i][j]++,lie[j]++;
24      for (int i=1;i<=n;i++)
25       for (int j=1;j<=m;j++){
26           if (sigh[i][j]){
27               if (f[i][j]==maxh[i]){
28                   if (hang[i]!=1)
29                       if (f[i][j]!=maxl[j])
30                           f[i][j]=1,hang[i]--,sigh[i][j]--;
31                       else if (lie[j]!=1)
32                           f[i][j]=1,lie[j]--,hang[i]--,sigh[i][j]-=2;
33               }
34               if (f[i][j]==maxl[j]){
35                   if (lie[j]!=1)
36                       if (f[i][j]!=maxh[i])
37                           f[i][j]=1,lie[j]--,sigh[i][j]--;
38                       else if (hang[i]!=1)
39                           f[i][j]=1,hang[i]--,lie[j]--,sigh[i][j]-=2;
40               }
41           }
42           else if (f[i][j]!=0) f[i][j]=1;
43     }
44     long long ans=0;
45     for (int i=1;i<=n;i++)
46      for (int j=1;j<=m;j++)
47          ans+=f[i][j];
48     printf("%lld\n",tmp-ans);
49     return 0;
50 }
贪心233

 

JZOJ.5307【NOIP2017模拟8.18】偷窃

标签:heap   wcry   bean   unp   avr   jni   psi   vdp   psk   

原文地址:http://www.cnblogs.com/Lanly/p/7391268.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!