题目: 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 且 使变后的序列 字典序 最小。
就可以跑匈牙利了(迷之时间复杂度。。。)
其实就是要判断是否 完美匹配 。若没有 完美匹配 ,则输出无解。
注意匈牙利要倒着跑,可以跑出 字典序 最小。
 #include<bits/stdc++.h>
using namespace std;
#define MAXN 10010
int BF[MAXN],BF1[MAXN],d[MAXN],prey[MAXN],n,f[MAXN][];
//bool f[MAXN][MAXN];
bitset<MAXN> vis;
int read()
{
int s=,fh=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')fh=-;ch=getchar();}
while(ch>=''&&ch<=''){s=s*+(ch-'');ch=getchar();}
return s*fh;
}
int Xyl(int u)
{
int i,v;
for(i=;i<=;i++)
{
v=f[u][i];
if(vis[v]==)
{
vis[v]=;
if(Xyl(BF[v])==||BF[v]==)
{
BF[v]=u;
BF1[u]=v;
return ;
}
}
}
return ;
}
int main()
{
freopen("transform.in","r",stdin);
freopen("transform.out","w",stdout);
int i,ans,dd,gs;
n=read();
memset(f,false,sizeof(f));
for(i=;i<n;i++)
{
d[i]=read();dd=n-d[i];
gs=-;
if(i-d[i]>=)f[i][++gs]=i-d[i];
if(i+d[i]<=n-)f[i][++gs]=i+d[i];
if(i-dd>=)f[i][++gs]=i-dd;
if(i+dd<=n-)f[i][++gs]=i+dd;
if(f[i][]>=f[i][])swap(f[i][],f[i][]);
}
memset(BF,,sizeof(BF));ans=;
memset(prey,-,sizeof(prey));
for(i=n-;i>=;i--)
{
vis.reset();
ans+=Xyl(i);
//prey[i]=BF1[i];
}
if(ans!=n)printf("No Answer");
else
{
//for(i=0;i<n;i++)BF1[BF[i]]=i;
for(i=;i<n;i++)printf("%d ",BF1[i]);
}
fclose(stdin);
fclose(stdout);
return ;
}

Bzoj 1562: [NOI2009]变换序列 匈牙利算法,二分图匹配的更多相关文章

  1. BZOJ 1562 [NOI2009] 变换序列

    [NOI2009] 变换序列 [题解] 就是有一个序列,每个位置可以填两个数,不可重复,问最小字典序. 显然,可以建一个二分图,判合法就是找完美匹配. 那怎么弄最小字典序呢?有好多种解法,我这里给出了 ...

  2. BZOJ 1562 [NOI2009]变换序列:二分图匹配

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1562 题意: 给定n,定义D(x,y) =  min(|x-y|, n-|x-y|),然后 ...

  3. 1562: [NOI2009]变换序列 - BZOJ

    Description Input Output Sample Input 5 1 1 2 2 1 Sample Output 1 2 4 0 3 HINT 30%的数据中N≤50:60%的数据中N≤ ...

  4. 1562. [NOI2009]变换序列【二分图】

    Description Input Output Sample Input 5 1 1 2 2 1 Sample Output 1 2 4 0 3 HINT 30%的数据中N≤50: 60%的数据中N ...

  5. Luogu P1963 [NOI2009]变换序列(二分图匹配)

    P1963 [NOI2009]变换序列 题意 题目描述 对于\(N\)个整数\(0,1, \cdots ,N-1\),一个变换序列\(T\)可以将\(i\)变成\(T_i\),其中\(T_i \in ...

  6. [模板] 匈牙利算法&&二分图最小字典序匹配

    匈牙利算法 简介 匈牙利算法是一种求二分图最大匹配的算法. 时间复杂度: 邻接表/前向星: \(O(n * m)\), 邻接矩阵: \(O(n^3)\). 空间复杂度: 邻接表/前向星: \(O(n ...

  7. [Luogu 1963] NOI2009 变换序列

    [Luogu 1963] NOI2009 变换序列 先%Dalao's Blog 什么?二分图匹配?这个确定可以建图? 「没有建不成图的图论题,只有你想不出的建模方法.」 建图相当玄学,不过理解大约也 ...

  8. noi2009变换序列

    noi2009变换序列 一.题目 1843 变换序列 2009年NOI全国竞赛  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 大师 Master 题解       题目描述  ...

  9. # 匈牙利算法(二分图最大匹配)- hdu 过山车

    匈牙利算法(二分图最大匹配)- hdu 过山车 Hdu 2063 二分图:图中的点可以分成两组U,V,所有边都是连接U,V中的顶点.等价定义是:含奇数条边的图. 匹配:一个匹配是一个边的集合,其中任意 ...

随机推荐

  1. 大话F#和C#:是否会重蹈C#失败的覆辙?

    F#.net 出来有些年头儿了,将从 VS 2010 起在 .net framework 平台上以“一等公民”身份粉墨登场的它,将会给计算机科技与软件工业带来哪些悲喜剧呢? F# 将扮演一个什么角色? ...

  2. How to open .ccproj in VS2010?

    Q: How to open .ccproj projects types in VS2010, ccproj file type is a Cloud project i suppose. Plea ...

  3. django 中的延迟加载技术,python中的lazy技术

    ---恢复内容开始--- 说起lazy_object,首先想到的是django orm中的query_set.fn.Stream这两个类. query_set只在需要数据库中的数据的时候才 产生db ...

  4. 补充:学会Twitter Bootstrap不再难

    博客园的兄弟姐妹们很给力,自从这篇文章写出后,有人可能会对2.x版本升级到3.x版本的区别有些好奇和模糊.现在将官方给出的说明贴上去: 从2.x升级到3.0版本 Bootstrap 3并不向后兼容Bo ...

  5. hdu 5626 Clarke and points 数学推理

    Clarke and points Problem Description   The Manhattan Distance between point A(XA,YA) and B(XB,YB) i ...

  6. Js通过原型继承创建子类

    //定义一个有两个方法的类 function Person(){} Person.prototype.married = function(){}; Person.prototype.unmerrie ...

  7. NPOI常用功能工具类

    public class NPOIHelper { /// <summary> /// DataTable导出到Excel文件 /// </summary> /// <p ...

  8. zoj 3761

    很简单但很虐心的一道题: 我感觉自己的算法没错,但是老是过不了:= = 但是还是把代码贴出来: 我的方法,用并查集的方式做一课树,然后对树进行遍历: 有多少棵树就有多少个点剩余: #include&l ...

  9. codeforces 392A Blocked Points

    我的方式是用暴力的方法找到每一行每一列的边界点: 但是有大神直接用一个公式解决了:floor(n*sqrt(2))*4: 想了很久还是不理解,求各路大神指点! #include<iostream ...

  10. Java集合框架的知识总结(1)

    说明:先从整体介绍了Java集合框架包含的接口和类,然后总结了集合框架中的一些基本知识和关键点,并结合实例进行简单分析. 1.综述 所有集合类都位于java.util包下.集合中只能保存对象(保存对象 ...