Bzoj 1562: [NOI2009]变换序列 匈牙利算法,二分图匹配

题目: http://cojs.tk/cogs/problem/problem.php?pid=409

409. [NOI2009]变换序列

★★☆   输入文件:transform.in   输出文件:transform.out   简单对比
时间限制:1 s   内存限制:128 MB

 【问题描述】

       对于 N个整数0, 1, ……,  N-1,一个变换序列 T可以将 i变成 Ti,其中
定义 xy之间的距离 。给定每个 iTi之间的距离 D( i, Ti),
你需要求出一个满足要求的变换序列 T。如果有多个满足条件的序列,输出其中字典序最小的一个。
 
说明:对于两个变换序列 ST,如果存在 p< N,满足对于 i=0,1,…… p-1, Si= TiSp< Tp,我们称 ST字典序小。
【输入文件】
       输入文件transform.in的第一行包含一个整数 N,表示序列的长度。接下来的一行包含 N个整数 Di,其中 Di表示 iTi之间的距离。
【输出文件】
       输出文件为transform.out。
如果至少存在一个满足要求的变换序列 T,则输出文件中包含一行 N个整数,表示你计算得到的字典序最小的 T;否则输出”No Answer”(不含引号)。 注意:输出文件中相邻两个数之间用一个空格分开,行末不包含多余空格。
【输入样例】
5
1 1 2 2 1
【输出样例】
1 2 4 0 3
【数据规模和约定】
20%的数据中 N≤50;
60%的数据中 N≤500;
100%的数据中 N≤10000。
 
题解:
二分图匹配+匈牙利算法
设原序列值为x[0],x[1],x[2]......x[n-1],要变成的序列值为y[0],y[1],y[2]......y[n-1].
原序列也就是0,1,2......n-1,即x[0]=0,x[1]=1,x[2]=2......x[n-1]=n-1.
首先,通过样例可以想到每一个值x[i]可以变成的 y[i]的取值最多只有两个,但为什么,证明如下:
 
设a=| x[i] - y[i] |,且a ≤ (n-a),其中a,(n-a)都为差值(由题目中 可知)
 
对于每一个x[i],我们有y[i]=x[i]±a或y[i]=x[i]±(n-a).
 
因为每个x[i]和y[i]要满足0≤x[i]≤n-1,0≤y[i]≤n-1,且y[i]=x[i]±a或y[i]=x[i]±(n-a).
 
则可以得到:  x[i]+a≤n-1   --------------   (1)式
                   x[i]-a≥0       --------------   (2)式
                   x[i]+(n-a)≤n-1  ----------    (3)式
                   x[i]-(n-a)≥0   -------------   (4)式
 
(3)式化简可得 : x[i]-a≤-1
 
(4)式移项可得 : x[i]+a≥n
 
观察四个式子,我们可以发现 :(1)式 和 (4)式 是矛盾的,(2)式 和(3)式 是矛盾的.
 
所以,我们只能在 (1)式 和 (4)式 中选一个,(2)式 和(3)式 中选一个.
 
证毕.
 
然后,有了这个结论,我们就把题目变成了 :给出0...n-1,还给出了每个数的变化量(变化量给的是绝对值,也就是可以变化的差值),每个数都必须要 变化 且 恰好变化给定的差值 ,每个数可以变成两个数(上面已经证明),让你 变成0...n-1 且 使变后的序列 字典序 最小。
就可以跑匈牙利了(迷之时间复杂度。。。)
其实就是要判断是否 完美匹配 。若没有 完美匹配 ,则输出无解。
注意匈牙利要倒着跑,可以跑出 字典序 最小。
 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 #define MAXN 10010
 4 int BF[MAXN],BF1[MAXN],d[MAXN],prey[MAXN],n,f[MAXN][2];
 5 //bool f[MAXN][MAXN];
 6 bitset<MAXN> vis;
 7 int read()
 8 {
 9     int s=0,fh=1;char ch=getchar();
10     while(ch<'0'||ch>'9'){if(ch=='-')fh=-1;ch=getchar();}
11     while(ch>='0'&&ch<='9'){s=s*10+(ch-'0');ch=getchar();}
12     return s*fh;
13 }
14 int Xyl(int u)
15 {
16     int i,v;
17     for(i=0;i<=1;i++)
18     {
19         v=f[u][i];
20         if(vis[v]==0)
21         {
22             vis[v]=1;
23             if(Xyl(BF[v])==1||BF[v]==0)
24             {
25                 BF[v]=u;
26                 BF1[u]=v;
27                 return 1;
28             }
29         }
30     }
31     return 0;
32 }
33 int main()
34 {
35     freopen("transform.in","r",stdin);
36     freopen("transform.out","w",stdout);
37     int i,ans,dd,gs;
38     n=read();
39     memset(f,false,sizeof(f));
40     for(i=0;i<n;i++)
41     {
42         d[i]=read();dd=n-d[i];
43         gs=-1;
44         if(i-d[i]>=0)f[i][++gs]=i-d[i];
45         if(i+d[i]<=n-1)f[i][++gs]=i+d[i];
46         if(i-dd>=0)f[i][++gs]=i-dd;
47         if(i+dd<=n-1)f[i][++gs]=i+dd;
48         if(f[i][0]>=f[i][1])swap(f[i][0],f[i][1]);
49     }
50     memset(BF,0,sizeof(BF));ans=0;
51     memset(prey,-1,sizeof(prey));
52     for(i=n-1;i>=0;i--)
53     {
54         vis.reset();
55         ans+=Xyl(i);
56         //prey[i]=BF1[i];
57     }
58     if(ans!=n)printf("No Answer");
59     else
60     {
61         //for(i=0;i<n;i++)BF1[BF[i]]=i;
62         for(i=0;i<n;i++)printf("%d ",BF1[i]);
63     }
64     fclose(stdin);
65     fclose(stdout);
66     return 0;
67 }

 

转载于:https://www.cnblogs.com/Var123/p/5399165.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值