[BZOJ1998][Hnoi2010]Fsk物品调度

[BZOJ1998][Hnoi2010]Fsk物品调度

试题描述

现在找工作不容易,Lostmonkey费了好大劲才得到fsk公司基层流水线操作员的职位。流水线上有n个位置,从0到n-1依次编号,一开始0号位置空,其它的位置i上有编号为i的盒子。Lostmonkey要按照以下规则重新排列这些盒子。 规则由5个数描述,q,p,m,d,s,s表示空位的最终位置。首先生成一个序列c,c0=0,ci+1=(ci*q+p) mod m。接下来从第一个盒子开始依次生成每个盒子的最终位置posi,posi=(ci+d*xi+yi) mod n,xi,yi是为了让第i个盒子不与之前的盒子位置相同的由你设定的非负整数,且posi还不能为s。如果有多个xi,yi满足要求,你需要选择yi最小的,当yi相同时选择xi最小的。 这样你得到了所有盒子的最终位置,现在你每次可以把某个盒子移动到空位上,移动后原盒子所在的位置成为空位。请问把所有的盒子移动到目的位置所需的最少步数。

输入

第一行包含一个整数t,表示数据组数。接下来t行,每行6个数,n,s,q,p,m,d意义如上所述。 对于30%的数据n<=100,对于100%的数据t<=20,n<=100000,s

输出

对于每组数据输出一个数占一行,表示最少移动步数。

输入示例

1
8 3 5 2 7 4

输出示例

6

数据规模及约定

t<=20,n<=100000

题解

刚刚在火车上调出了这道题。。。

关键在于如何求 pos 数组,不难发现式子中 d 是固定的,所以 xi 每增加 1,posi 就要增加 d,而 yi 每增加 1,posi 会增加 1.

再看看题目要求优先考虑令 yi 最小,即,固定 yi,改变 xi.假设我们已经把 yi 固定下来了,则可以将 1~n 的数按照对 d 取余得到的余数分类,通过 ci 和固定下来的 yi 的值确定要找的数在哪类,选取没有被选过的且离 ci “向右距离”(从 ci 出发向右走,遇到 n 就回到 0,继续向右到达该数所需的步数)最近的那个数就行了,可以用个并查集实现。那么如何固定 yi 呢?类似地,也可以对于每一数建立一个并查集,当某一类数都被选取后,将该节点与左右合并,查找时找没有满的且离 yi + ci 所属的类“向右距离”最近的一类即可。

最后求步数不妨放自己yy。

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;

#define maxn 100010
#define LL long long
int n, s, q, p, m, d, gcdnd, pos[maxn];
bool has[maxn], h2[maxn];

int fa[maxn], siz[maxn];
int findset(int x) { return x == fa[x] ? x : fa[x] = findset(fa[x]); }
int f2[maxn];
int find2(int x) { return x == f2[x] ? x : f2[x] = find2(f2[x]); }
int nxt(int x) { return (x + d) % n; }
int pre(int x) { return (x - d + n) % n; }
LL gcd(LL a, LL b) { return !b ? a : gcd(b, a % b); }
void add(int u, int x) {
	has[u] = 1;
	int v = findset(nxt(u));
	if(has[v]) fa[u] = v;
	v = findset(pre(u)); u = findset(u);
	if(u != v) { if(has[v]) fa[v] = u; }
	else {
		h2[x] = 1; int xx = find2(x);
		if(x < gcdnd - 1 && h2[x+1]){ v = find2(x + 1); if(x != v) f2[x] = v; }
		if(x && h2[x-1]){ v = find2(x - 1); if(x != v) f2[v] = x; }
	}
	return ;
}

int main() {
	int T; scanf("%d", &T);
	while(T--) {
		scanf("%d%d%d%d%d%d", &n, &s, &q, &p, &m, &d); d %= n;
//		n = read(); s = read(); q = read(); p = read(); m = read(); d = read() % n;
		gcdnd = gcd(n, d);
		memset(has, 0, sizeof(has));
		memset(h2, 0, sizeof(h2));
		has[s] = 1;
		for(int i = 0; i < n; i++) fa[i] = i;
		for(int i = 0; i < gcdnd; i++) f2[i] = i;
		LL c = 0;
		for(int i = 1; i < n; i++) {
			c = (c * q + p) % m;
			LL cd = c % n % gcdnd;
			int y = find2(cd); if(h2[y]) y++; if(y >= gcdnd){ y = find2(0); if(h2[y]) y++; }
			int u;
			if(y - cd >= 0){ u = findset((c % n + y - cd) % n); if(has[u]) u = nxt(u); }
			else { u = findset((c % n + y - cd + gcdnd) % n); if(has[u]) u = nxt(u); }
			add(u, y); pos[i] = u;
		}
		 
//		for(int i = 1; i < n; i++) printf("%d ", pos[i]); puts("\n");
		for(int i = 0; i < n; i++) fa[i] = i, siz[i] = 1;
		for(int i = 1; i < n; i++) {
			int u = findset(i), v = findset(pos[i]);
			if(u != v) fa[v] = u, siz[u] += siz[v], siz[v] = 0;
		}
		for(int i = 0; i < n; i++) pos[i] = findset(i);
		sort(pos, pos + n);
		int ans = 0;
		for(int i = 0; i < n; i++) if((!i || pos[i] != pos[i-1]) && siz[pos[i]] > 1) ans += siz[pos[i]] + 1;
		if(siz[findset(0)] > 1) ans -= 2;
		printf("%d\n", ans);
	}
	 
	return 0;
}

 

posted @ 2016-06-29 13:26  xjr01  阅读(218)  评论(0编辑  收藏  举报