热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

[NOI2009变换序列]

[关键字]:图论二分图匹配[题目大意]:太麻烦,就不说了[分析]:因为每个距离都对应着两个数字——就是d或-d,

[关键字]:图论 二分图匹配

[题目大意]:太麻烦,就不说了

//=========================================================================

[分析]:因为每个距离都对应着两个数字——就是+d或-d,这样0~n-1和它所转换的序列可以看成一个二分图的模型,而一个完备匹配就对应着一个序列。只要求一遍最大匹配看是否等于n。而字典序的问题,因为用匈牙利算法找增广路就是找到第一个没有被匹配或和它匹配的点可以找到另外一条增广路的点,这样每次都是将已经匹配过的点所匹配的点编号变大,如果倒着找匹配就恰好能保证序号越小就越可以找到编号小的点匹配。

[代码]:

View Code

#include
#include
#include
#include
#include
#include
using namespace std;

const int MAXN=20000;

int n;
int m1[MAXN],m2[MAXN];
bool v[MAXN];
vector<int> e[MAXN];

void Init()
{
int t1,t2,d;
scanf("%d",&n);
for (int i&#61;0;i {
scanf("%d",&d);
t1&#61;i&#43;d;
if (t1>&#61;n) t1%&#61;n;
t2&#61;i-d;
if (t2<0) t2&#43;&#61;n;
if (t1>t2) swap(t1,t2);
e[i].push_back(t1),e[i].push_back(t2);
}
}

bool DFS(int u)
{
for (vector<int>::iterator i&#61;e[u].begin();iif (!v[*i])
{
v[*i]&#61;1;
if (m2[*i]&#61;&#61;-1 || DFS(m2[*i]))
{
m1[u]&#61;*i;
m2[*i]&#61;u;
return 1;
}
}
return 0;
}

void Solve()
{
memset(m1,255,sizeof(m1));
memset(m2,255,sizeof(m2));
int ans&#61;0;
//for (int i&#61;0;i
for (int i&#61;n-1;i>&#61;0;--i)
{
memset(v,0,sizeof(v));
if (DFS(i)) &#43;&#43;ans; else break;
}
if (ans!&#61;n) printf("No Answer\n");
else
{
printf("%d",m1[0]);
for (int i&#61;1;i"
%d",m1[i]);
}
}

int main()
{
Init();
Solve();
system("pause");
return 0;
}



转:https://www.cnblogs.com/procedure2012/archive/2012/03/16/2400834.html



推荐阅读
author-avatar
wo缘相聚在空间
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有