篝火晚会

描述

佳佳刚进高中,在军训的时候,由于佳佳吃苦耐劳,很快得到了教官的赏识,成为了“小教官”。在军训结束的那天晚上,佳佳被命令组织同学们进行篝火晚会。一共有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。我们需要佳佳用最少的总代价实现同学们的意愿,你能帮助佳佳吗?

对于30%的数据,n <= 1000;

对于全部的数据,n <= 50000。

格式

输入格式

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

输出格式

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

样例1

样例输入1[复制]

4
3 4
4 3
1 2
1 2

样例输出1[复制]

2

限制

1s

来源

NOIp2005 第三题

  注意:题目没有要求b1,b2,b3......bM是一段连续的序列。

  根据群论原理,任何置换群都可以分解为若干个循环节。显然,这道题目就是冲着这点来的。因为如果某循环的长度为L,那么本题中只需要代价为L的操作。注意,这里的L!=1,这是因为只有一个人的循环节不需要任何代价的操作。到这里才只能拿三十分,因为圈是可以旋转的,常规方法需要O(n^2)才能解决。不妨这里以编号为1的人为基准,定义每个人到自己应该所在位置的距离。距离不超过n-1。可以通过最大值来寻找能在自己位置上的最多的人数(因为距离为x的人在旋转x次后转到自己的位置上),那么在比较好的情况下,可以优化到O(n)。

  http://wenku.baidu.com/view/878beb64783e0912a2162aa7.html?qq-pf-to=pcqq.c2c

  30暴力

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstdlib>
 4 #include<cmath>
 5 #include<algorithm>
 6 using namespace std;
 7 typedef long long LL;
 8 const int MAX_N=50005;
 9 struct node{
10     int x,y;
11 }stu[MAX_N];
12 int N;
13 int pos[MAX_N];
14 int tar[MAX_N];
15 int vis[MAX_N];
16 int t1[MAX_N];
17 int t2[MAX_N];
18 int ANS;//记录不需要移动的个数
19 void make_tar();
20 void move();
21 void calc();
22 int main(){
23     scanf("%d",&N);
24     for(int i=1;i<=N;i++){
25         scanf("%d%d",&stu[i].x,&stu[i].y);
26         pos[i]=i;
27     }
28
29     make_tar();
30     calc();
31
32     for(int i=1;i<=N-1;i++){
33         move();
34         calc();
35     }
36     cout<<N-ANS;
37     return 0;
38
39 }
40 void make_tar(){// 生成目标序列
41     tar[1]=1; vis[tar[1]]=1;
42     tar[N]=stu[1].x; vis[tar[N]]=1;
43     tar[2]=stu[1].y; vis[tar[2]]=1;
44     int now=2;
45     while(now!=N){
46         int l=stu[tar[now]].x;
47         int r=stu[tar[now]].y;
48         if(tar[now-1]!=l){
49             tar[now+1]=l;
50             vis[tar[now+1]]=1;
51         }
52         else if(tar[now-1]!=r){
53             tar[now+1]=r;
54             vis[tar[now+1]]=1;
55         }
56         else{
57             cout<<-1;
58             exit(0);
59         }
60         now++;
61     }
62     for(int i=1;i<=N;i++){
63         if(vis[i]!=1){
64             cout<<-1;
65             exit(0);
66         }
67     }
68
69 }
70 void move(){
71     int now=1;
72     int temp1=pos[now];
73     while(now!=N){
74         int temp2=pos[now+1];
75         pos[now+1]=temp1;
76         temp1=temp2;
77         now++;
78     }
79     pos[1]=temp1;
80 }
81 void calc(){
82     int ans=0;
83     for(int i=1;i<=N;i++){
84         if(pos[i]==tar[i]){
85             ans++;
86         }
87     }
88     ANS=max(ANS,ans);
89 }

  100分AC

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstdlib>
 4 #include<cmath>
 5 #include<algorithm>
 6 using namespace std;
 7 typedef long long LL;
 8 const int MAX_N=50005;
 9 struct node{
10     int x,y;
11 }stu[MAX_N];
12 int N;
13 int pos[MAX_N];
14 int tar[MAX_N];
15 int vis[MAX_N];
16 int t1[MAX_N];
17 int t2[MAX_N];
18 int ANS;//记录不需要移动的个数
19 void make_tar();
20 void move();
21 void calc();
22 int main(){
23     scanf("%d",&N);
24     for(int i=1;i<=N;i++){
25         scanf("%d%d",&stu[i].x,&stu[i].y);
26         pos[i]=i;
27     }
28
29     make_tar();
30     for(int i=1;i<=N;i++){
31         int t=(tar[i]-i+N)%N;
32         t1[t]++;
33         ANS=max(ANS,t1[t]);
34         t=(tar[N-i+1]-i+N)%N;
35         t2[t]++;
36         ANS=max(ANS,t2[t]);
37     }
38     cout<<N-ANS;
39     return 0;
40
41 }
42 void make_tar(){// 生成目标序列
43     tar[1]=1; vis[tar[1]]=1;
44     tar[N]=stu[1].x; vis[tar[N]]=1;
45     tar[2]=stu[1].y; vis[tar[2]]=1;
46     int now=2;
47     while(now!=N){
48         int l=stu[tar[now]].x;
49         int r=stu[tar[now]].y;
50         if(tar[now-1]!=l){
51             tar[now+1]=l;
52             vis[tar[now+1]]=1;
53         }
54         else if(tar[now-1]!=r){
55             tar[now+1]=r;
56             vis[tar[now+1]]=1;
57         }
58         else{
59             cout<<-1;
60             exit(0);
61         }
62         now++;
63     }
64     for(int i=1;i<=N;i++){
65         if(vis[i]!=1){
66             cout<<-1;
67             exit(0);
68         }
69     }
70
71 }
时间: 2024-10-11 11:39:58

篝火晚会的相关文章

NOIP2005 篝火晚会

题目描述 Description 佳佳刚进高中,在军训的时候,由于佳佳吃苦耐劳,很快得到了教官的赏识,成为了“小教官”.在军训结束的那天晚上,佳佳被命令组织同学们进行篝火晚会.一共有n个同学,编号从1到n.一开始,同学们按照1,2,……,n的顺序坐成一圈,而实际上每个人都有两个最希望相邻的同学.如何下命令调整同学的次序,形成新的一个圈,使之符合同学们的意愿,成为摆在佳佳面前的一大难题.佳佳可向同学们下达命令,每一个命令的形式如下:(b1,b2,...bm-1,bm)这里m的值是由佳佳决定的,每次

NOIP2005 篝火晚会 解题报告

佳佳刚进高中,在军训的时候,由于佳佳吃苦耐劳,很快得到了教官的赏识,成为了“小教官”.在军训结束的那天晚上,佳佳被命令组织同学们进行篝火晚会.一共有n个同学,编号从1到n.一开始,同学们按照1,2,……,n的顺序坐成一圈,而实际上每个人都有两个最希望相邻的同学.如何下命令调整同学的次序,形成新的一个圈,使之符合同学们的意愿,成为摆在佳佳面前的一大难题. 佳佳可向同学们下达命令,每一个命令的形式如下: (b1, b2,... bm -1, bm) 这里m的值是由佳佳决定的,每次命令m的值都可以不同

【codevs1106】 篝火晚会

http://codevs.cn/problem/1106/ (题目链接) 题意 将1~n顺序排列的环改成另一个环,问n-不动点数. Solution 啊智障啦,不会做×_× 左转hzwer 代码 // codevs1106 #include<algorithm> #include<iostream> #include<cstdlib> #include<cstring> #include<cstdio> #include<cmath>

Vijos1008 篝火晚会

题目大意:给定初始序列和构建目标序列的条件,问最少需要多少步所谓的"变换"能达成该序列. 对题目的理解:如果按照原文所说,"变换"的定义是"这个命令的作用是移动编号是b1,b2,-- bm–1,bm的这m个同学的位置.要求b1换到b2的位置上,b2换到b3的位置上,--,要求bm换到b1的位置上",但这条命令究竟是什么意思呢?我相信不少人的理解是将a1至am这些数字循环移动一位(实际上我刚开始也是这样想的,后来发现这样想越来越诡异,便看了题解),

[vijos P1008 篝火晚会]置换

题意:编号1-n的小朋友依次围成一圈,给定目标状态每个小朋友左右两边的小朋友编号,每次可以选择编号为[b1,b2,...,bm]的小朋友,作1次轮换,bi是任意编号,代价为m.求变成目标状态所需的最小代价. 思路:有置换的知识,任意一个置换可以写成若干循环的乘积,那么每次选择一个大小大于1的循环,把这个循环变成目标状态,代价为循环的大小.那么要使总代价最小,就要使得大小大于1的循环的大小和最小,也就是大小为1的循环的个数尽量多,也就是把目标状态和原状态进行比较,使得对应位置相等的点最多.现在来求

【置换群/模拟】NOIP2005-篝火晚会

[问题描述] 佳佳刚进高中,在军训的时候,由于佳佳吃苦耐劳,很快得到了教官的赏识,成为了“小教官”.在军训结束的那天晚上,佳佳被命令组织同学们进行篝火晚会.一共有n个同学,编号从1到n.一开始,同学们按照1,2,……,n的顺序坐成一圈,而实际上每个人都有两个最希望相邻的同学.如何下命令调整同学的次序,形成新的一个圈,使之符合同学们的意愿,成为摆在佳佳面前的一大难题. 佳佳可向同学们下达命令,每一个命令的形式如下: (b1, b2,... bm -1, bm) 这里m的值是由佳佳决定的,每次命令m

20170730 《战狼2》观影纪要

去看<战狼2>的那天,其实本来是打算去看<绣春刀2>的,但是没票了,听说<战狼2>口碑不错,就改了.那天没想到<战狼2>会是一部50亿+票房的主. 由于看完已经两周了,下面仅凭记忆提取下内容概要. 开局就是水下激战的那一段,感觉还行,可能拍的挺辛苦,但是个人感觉没有网络上说的那么震撼.见仁见智了. 观众不禁会想,为什么冷锋跑到一个商用船上,而且还是在非洲.回到三年前,一个拆迁现场,冷锋几个战友带着上一级中牺牲的队长骨灰回到老家,结果战友的老家正在被一个包工头

小测(noip2005的两道题) 2017.3.3

过河 题目描述 Description 在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧.在桥上有一些石子,青蛙很讨厌踩在这些石子上.由于桥的长度和青蛙一次跳过的距离都是正整数,我们可以把独木桥上青蛙可能到达的点看成数轴上的一串整点:0,1,……,L(其中L是桥的长度).坐标为0的点表示桥的起点,坐标为L的点表示桥的终点.青蛙从桥的起点开始,不停的向终点方向跳跃.一次跳跃的距离是S到T之间的任意正整数(包括S,T).当青蛙跳到或跳过坐标为L的点时,就算青蛙已经跳出了独木桥.题目给出独

求虐,我还想再拓展一次

前言:孟子曰:"天将降大任于斯人也,必先苦其心志,劳其筋骨,饿其体肤,空伐其身,行弗乱其所为,所以动心忍性,曾益其所不能."那么两天一夜的"洛阳兄弟连拓展训练"让我找到了久违的受虐感觉,纵使休息了一夜的我依然身体欠恙,然而内心却不再叫苦不迭,更多的是,感谢领导的睿智,给了我们全体员工的锻炼机会.所以我想大声的呐喊:"求虐,我还想再拓展一次!" 回忆总是美好的,因为什么呢?其实我也说不上缘由,哈哈. PS:这是一张野外休息时我们项目部部分成员的照片