[vijos P1008 篝火晚会]置换

题意:编号1-n的小朋友依次围成一圈,给定目标状态每个小朋友左右两边的小朋友编号,每次可以选择编号为[b1,b2,...,bm]的小朋友,作1次轮换,bi是任意编号,代价为m。求变成目标状态所需的最小代价。

思路:有置换的知识,任意一个置换可以写成若干循环的乘积,那么每次选择一个大小大于1的循环,把这个循环变成目标状态,代价为循环的大小。那么要使总代价最小,就要使得大小大于1的循环的大小和最小,也就是大小为1的循环的个数尽量多,也就是把目标状态和原状态进行比较,使得对应位置相等的点最多。现在来求这个最大值,由于目标状态是个环,起点有n个,有2个方向,所以总共2n种不同状态,但是状态之间是有联系的。先考虑一个方向,预处理出某个状态与原状态对应位上的差,此时的答案就是差为0的个数,当起点变为下一个时,后n-1个位(或前n-1个位)的差全部变化为1,剩下的一位特殊处理。于是可以设计一个这样的数据结构,用一个mark标记表示当前区间整体变化了多少,cnt[i+mark]就表示当前值为i的个数,每次取cnt[mark]得到差为0的个数,更新时也通过mark标记定位到正确地址。复杂度O(n)。

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
#pragma comment(linker, "/STACK:10240000")
#include <map>
#include <set>
#include <cmath>
#include <ctime>
#include <deque>
#include <queue>
#include <stack>
#include <vector>
#include <cstdio>
#include <string>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>

using namespace std;

#define X                   first
#define Y                   second
#define pb                  push_back
#define mp                  make_pair
#define all(a)              (a).begin(), (a).end()
#define fillchar(a, x)      memset(a, x, sizeof(a))
#define copy(a, b)          memcpy(a, b, sizeof(a))

typedef long long ll;
typedef pair<int, int> pii;
typedef unsigned long long ull;

#ifndef ONLINE_JUDGE
void RI(vector<int>&a,int n){a.resize(n);for(int i=0;i<n;i++)scanf("%d",&a[i]);}
void RI(){}void RI(int&X){scanf("%d",&X);}template<typename...R>
void RI(int&f,R&...r){RI(f);RI(r...);}void RI(int*p,int*q){int d=p<q?1:-1;
while(p!=q){scanf("%d",p);p+=d;}}void print(){cout<<endl;}template<typename T>
void print(const T t){cout<<t<<endl;}template<typename F,typename...R>
void print(const F f,const R...r){cout<<f<<", ";print(r...);}template<typename T>
void print(T*p, T*q){int d=p<q?1:-1;while(p!=q){cout<<*p<<", ";p+=d;}cout<<endl;}
#endif
template<typename T>bool umax(T&a, const T&b){return b<=a?false:(a=b,true);}
template<typename T>bool umin(T&a, const T&b){return b>=a?false:(a=b,true);}

const double PI = acos(-1.0);
const int INF = 1e9 + 7;
const double EPS = 1e-12;

/* -------------------------------------------------------------------------------- */

const int maxn = 2e5 + 7;

struct Node {
    int a[maxn * 2];
    int &operator [] (int x) {
        return a[x + maxn];
    }
};

int l[maxn], r[maxn], b[maxn], c[maxn];
Node cnt;
bool vis[maxn];

int f(int id2, int id1) {
    return l[id1] == id2? r[id1] : l[id1];
}

int work(int a[], int n) {
    fillchar(cnt.a, 0);
    for (int i = 1; i <= n; i ++) {
        cnt[i - a[i]] ++;
    }
    int ans = cnt[0], mark = 0;
    for (int i = 1; i < n; i ++) {
        mark ++;
        cnt[i - a[i] + mark] --;
        cnt[n - a[i] + mark] ++;
        umax(ans, cnt[mark]);
    }
    return ans;
}

int main() {
//#ifndef ONLINE_JUDGE
//    freopen("in.txt", "r", stdin);
//    //freopen("out.txt", "w", stdout);
//#endif // ONLINE_JUDGE
    int n;
    cin >> n;
    for (int i = 1; i <= n; i ++) {
        scanf("%d%d", l + i, r + i);
    }
    b[1] = c[1] = 1;
    b[2] = l[1];
    c[2] = r[1];
    vis[1] = vis[l[1]] = true;
    for (int i = 3; i <= n; i ++) {
        b[i] = f(b[i - 2], b[i - 1]);
        c[i] = f(c[i - 2], c[i - 1]);
        vis[b[i]] = true;
    }
    for (int i = 1; i <= n; i ++) {
        if (!vis[i]) {
            puts("-1");
            return 0;
        }
    }
    int ans = 0;
    umax(ans, work(b, n));
    umax(ans, work(c, n));
    cout << n - ans << endl;
    return 0;
}
时间: 2024-11-10 07:55:50

[vijos P1008 篝火晚会]置换的相关文章

篝火晚会

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

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这些数字循环移动一位(实际上我刚开始也是这样想的,后来发现这样想越来越诡异,便看了题解),

【置换群/模拟】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:这是一张野外休息时我们项目部部分成员的照片