9.22考试 crf的军训 题解

  做这道题时由于第一道题太水了,第一反应是NOIP T2级别的题,需要拿上70~100的分,然后就开始分析,当然最后事实证明我错了……

  这道题当时首先联想到了 NOIP2016愤怒的小鸟 当然,数据范围不允许,但是我当时只是为了先拿到小数据的分数,所以先没考虑数据范围,在这里简单提一下:首先我先枚举了每一个状态,然后判断这个状态中的书是否能连在一起,然后就是一个2^(2*n)的转移,好吧,我承认,不是正宗的愤怒的小鸟打法,是当时集中生智编出来的,但是对于n<=10的复杂度还是够用的。

  然后对于小于80(我自己假定的一个数据范围)打了一个小爆搜,挂了。

  对于大于80的,也就是对应的正解部分,我首先想到的是贪心,然而被我自己rand出来的数据和状压一对比卡掉了,然后以为是动归,想了半天也没写出转移方程,于是也跪了,然后就开始往图论那里想,却一无所获。但是由于前一天刚刚看完《骗分导论》,想着万一有一些点仁义的不去卡我贪心呢?于是就把贪心交了上去,结果全WA了,拿到数据后输出了贪心答案-2得了60分,事后想一想也是,贪心一定比正确答案只多不少,减去一些数可能就是正解。但当时谁能想到呢?

 


 

                   蒟蒻与正解的分割线                      


 

  现在我来说正解,二分图最小路径(链)覆盖,由于造数据的标程有误,我就只说最正确的了(在此鸣谢撸串神的发现)。

  我们在二分图上跑最小链覆盖有一个巨大的前提:这个图一定是DAG,然而此题由于是要求“不超过”,所以会有两个规格一模一样的书之间有环,不过由于两本相同的书一定可以放在一起,所以我们只要把他们缩一下就行。

  然后由于最小链覆盖=n-最大匹配,所以我们只要向能放在他之后的点建边即可,然后就是最大匹配了。

 1 #include<iostream>
 2 #include<cstdlib>
 3 #include<cstdio>
 4 #include<cstring>
 5 #include<queue>
 6 #include<algorithm>
 7 #include<cmath>
 8 #include<map>
 9 #include<vector>
10 #define N 305
11 using namespace std;
12 int n;
13 struct no
14 {
15     int x,y;
16 }node[N],node2[N];
17 int px(no a,no b)
18 {
19     if(a.x==b.x)return a.y<b.y;
20     return a.x<b.x;
21 }
22 struct ro
23 {
24     int to,next;
25 }road[N*N*10];
26 int zz,a[N];
27 void build(int x,int y)
28 {
29     zz++;
30     road[zz].next=a[x];
31     road[zz].to=y;
32     a[x]=zz;
33 }
34 bool fw[N];
35 int b[3*N];
36 bool find(int x)
37 {
38     for(int i=a[x];i>0;i=road[i].next)
39     {
40         int y=road[i].to;
41         if(!fw[y])
42         {
43             fw[y]=1;
44             if(!b[y]||find(b[y]))
45             {
46                 b[y]=x;
47                 return 1;
48             }
49         }
50     }
51     return 0;
52 }
53 int main()
54 {
55     scanf("%d",&n);
56     for(int i=1;i<=n;i++)
57     {
58         scanf("%d%d",&node2[i].x,&node2[i].y);
59     }
60     sort(node2+1,node2+n+1,px);
61     int zzh=0;
62     for(int i=1;i<=n;i++)
63     {
64         if(node2[i].x!=node[zzh].y||node2[i].y!=node[zzh].y)
65         {
66             zzh++;
67             node[zzh]=node2[i];
68         }
69     }
70     n=zzh;
71     for(int i=1;i<=n;i++)
72     {
73         for(int j=1;j<=n;j++)
74         {
75             if(i==j)continue;
76             if(node[i].x<=node[j].x&&node[i].y<=node[j].y)
77                 build(i,j);
78         }
79     }
80     int ans=0;
81     for(int i=1;i<=n;i++)
82     {
83         memset(fw,0,sizeof(fw));
84         if(find(i))
85             ans++;
86 
87     }
88     printf("%d\n",n-ans);
89     return 0;
90 }
View Code

转载于:https://www.cnblogs.com/liutianrui/p/7581857.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值