Time Limit: 1 second

Memory Limit: 50 MB

【问题描述】

佳佳刚进高中,在军训的时候,由于佳佳吃苦耐劳,很快得到了教官的赏识,成为了“小教官”。在军训结束的那天晚上,佳佳被命令组织同学们进行篝火晚会。一共有n个同学,编号从1到n。一开始,同学们按照1,2,……,n的顺序坐成一圈,而实际上每个人都有两个最希望相邻的同学。如何下命令调整同学的次序,形成新的一个圈,使之符合同学们的意愿,成为摆在佳佳面前的一大难题。 佳佳可向同学们下达命令,每一个命令的形式如下: (b1, b2,... bm -1, bm) 这里m的值是由佳佳决定的,每次命令m的值都可以不同。这个命令的作用是移动编号是b1,b2,…… bm –1,bm的这m个同学的位置。要求b1换到b2的位置上,b2换到b3的位置上,……,要求bm换到b1的位置上。 执行每个命令都需要一些代价。我们假定如果一个命令要移动m个人的位置,那么这个命令的代价就是m。我们需要佳佳用最少的总代价实现同学们的意愿,你能帮助佳佳吗?

【输入】

共n+1行;
第一行是一个整数n(3 <= n <= 50000),表示一共有n个同学。
每行包括两个不同的正整数,以一个空格隔开,分别表示编号是1的同学最希望相邻的两个同学的编号,编号是2的同学最希望相邻的两个同学的编号,……,编号是n的同学最希望相邻的两个同学的编号。

【输出】

包含1行,这一行只包含一个整数,为最小的总代价。如果无论怎么调整都不能符合每个同学的愿望,则输出-1。

【输入样例】

4
3 4
4 3
1 2
1 2

【输出样例】

2

【 数据规模】

30%的数据满足:n <= 1000;
100%的数据满足:n <= 50000。

【题目链接】:http://noi.qz5z.com/viewtask.asp?id=b503

【题解】



首先,需要通过给出的n个人的需求;

判断出每个人的位置;

这里根据所给的关系,得出每个人的度数;

显然,构成一个环的要求是,每个人的度数都为2;

但是这样还不够,还得n个点都是联通的才行;不然两个独立的环

每个人的度数也都是2,但是不符合要求的;

然后根据输入得出这张换的一维形式;

即a[i]=t;表示第i个位置的人是t;

又一开始a[i]=i;

则先找出一开始不在自己位置上的人的个数y;

根据那个规则可以猜测y就是答案了;

但是这里一个环转换成一维的形式有N种;

你得算出每一种,不在自己位置上的人的个数;

然后求出最小值;

这里有一个思维的过程;

即先求出在自己位置上的人的最大个数的序列;

然后用n减去这个最大个数;

剩下的就是不在自己位置上的人数最小的人数了;

这个在自己位置上的人的最大个数可以通过偏移量来算;



随便选一个一维序列

b[1],b[2]…b[n];

表示第i个人要到哪一个位置;

然后对于每一个人

dic[b[i]-i]++;

这就表示第i个人从它原来的位置到它想要的位置需要把整个序列移动多少;

然后找最大的dic;

这个dic就是在自己想要在的位置上的人数的最大值;

因为有x个人他需要偏移dic[x]个位置就能回到自己的位置上,那么就整体把这个序列偏移dic[x]个位置就好;

假设N=7



向左偏移2格就相当于向右移动5格;

所以对于b[i]-i小于0的情况,直接加上n就好

即dic[(b[i]-i+n)%n]++;

然后再把这个随便选的环化线的序列b翻转一下(用reverse函数倒序一下就好);

再做一下上述过程;

因为我们只是左右偏移,还没办法覆盖翻转的情况;

所以还要再翻转一下才能把所有的情况都考虑到;

最后用n减去那个值就是答案了;



【完整代码】

#include <map>
#include <cmath>
#include <cstdio>
#include <algorithm>
using namespace std;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define LL long long
#define rep1(i,a,b) for (int i = a;i <= b;i++)
#define rep2(i,a,b) for (int i = a;i >= b;i--)
#define mp make_pair
#define pb push_back
#define fi first
#define se second
#define rei(x) scanf("%d",&x)
#define rel(x) scanf("%I64d",&x) typedef pair<int,int> pii;
typedef pair<LL,LL> pll; const int dx[9] = {0,1,-1,0,0,-1,-1,1,1};
const int dy[9] = {0,0,0,-1,1,-1,1,-1,1};
const double pi = acos(-1.0);
const int MAXN = 50000+100; int a[MAXN][2],du[MAXN],b[MAXN],cnt,n;
bool bo[MAXN];
map <int,int> dic; int main()
{
//freopen("F:\\rush.txt","r",stdin);
rei(n);
rep1(i,1,n)
{
rei(a[i][0]),rei(a[i][1]);
du[a[i][0]]++,du[a[i][1]]++;
}
rep1(i,1,n)
if (du[i]!=2)
{
puts("-1");
return 0;
}
int now = cnt = 1;
b[1] = now;
bo[1] = true;
while (cnt < n)
{
int x = a[now][0];
if (bo[x])
x = a[now][1];
if (bo[x])//如果往两个方向走都是走过的点
{//而且还没有走满n个点,就说明有多个独立的环
puts("-1");
return 0;
}
bo[x] = true;
now = x;
b[++cnt] = x;
}
rep1(i,1,n)
dic[(b[i]-i+n)%n]++;
int num = 0;
rep1(i,0,n-1)
num = max(num,dic[i]);
dic.clear();
reverse(b+1,b+1+n);
rep1(i,1,n)
dic[(b[i]-i+n)%n]++;
rep1(i,0,n-1)
num = max(num,dic[i]);
printf("%d\n",n-num);
return 0;
}

最新文章

  1. [WCF]缺少一行代码引发的血案
  2. CSS3 特效分解一
  3. 机器码call和jmp地址的计算
  4. 关于 c# 操作 world
  5. 【转】十分详细的xStream解析
  6. MFC 最大化 的时候控件 按比例变大
  7. 菜鸟学习Spring——60s学会Spring与Hibernate的集成
  8. Jsoup 的认识和简单使用
  9. Hitting the 2100 parameter limit (SQL Server) when using Contains()
  10. 如何排版 微信公众号「代码块」之 MarkEditor
  11. MongDB系列(一):使用node.js连接数据库
  12. 计算机网络之文件传送协议FTP
  13. docker入门实例
  14. Jenkins入门之执行定时任务
  15. C_数据结构_循环实现求阶乘
  16. webpack 4:默认配置
  17. 【SpringAop】【统一日志处理】注解方式理解以及使用
  18. 软件工程第二次作业(JUnit的使用)
  19. ueditor的上传文件漏洞(c#)
  20. python 视频 图像帧提取

热门文章

  1. 【JZOJ4747】【NOIP2016提高A组模拟9.3】被粉碎的线段树
  2. 杨柳目-杨柳科-Info-新闻:注意了!杨絮解决有办法了
  3. ios开发――解决UICollectionView的cell间距与设置不符问题
  4. MaxCompute 费用暴涨之新增SQL分区裁剪失败
  5. linux下安装composer以及使用composer安装laravel
  6. windows7蓝屏0x000000c4
  7. vue单页面项目返回上一页无效,链接变化了,但是页面没有变化
  8. VSCode 设置 CPP 代码风格
  9. 你真的知道你看到的UTF-8字符是什么吗?
  10. Oracle dbms_random包的用法