列表

详情


NC16899. [NOI2002]荒岛野人

描述

克里特岛以野人群居而著称。岛上有排列成环行的M个山洞。这些山洞顺时针编号为1,2,…,M。岛上住着N个野人,一开始依次住在山洞C1,C2,…,CN中,以后每年,第i个野人会沿顺时针向前走Pi个洞住下来。每个野人i有一个寿命值Li,即生存的年数。下面四幅图描述了一个有6个山洞,住有三个野人的岛上前四年的情况。三个野人初始的洞穴编号依次为1,2,3;每年要走过的洞穴数依次为3,7,2;寿命值依次为4,3,1。


奇怪的是,虽然野人有很多,但没有任何两个野人在有生之年处在同一个山洞中,使得小岛一直保持和平与宁静,这让科学家们很是惊奇。他们想知道,至少有多少个山洞,才能维持岛上的和平呢?

输入描述

第1行为一个整数N(1≤N≤15),即野人的数目。
第2行到第N+1每行为三个整数Ci, Pi, Li (1≤Ci,Pi≤100, 0≤Li≤106 ),表示每个野人所住的初始洞穴编号,每年走过的洞穴数及寿命值

输出描述

仅包含一个数M,即最少可能的山洞数。输入数据保证有解,且M不大于106

示例1

输入:

3
1 3 4
2 7 3
3 2 1

输出:

6

原站题解

上次编辑到这里,代码来自缓存 点击恢复默认模板

C++11(clang++ 3.9) 解法, 执行用时: 283ms, 内存消耗: 468K, 提交时间: 2020-02-14 20:59:13

#include<cstdio>
#include<algorithm>
const int N=20;
int n,s[N],p[N],l[N];
int exgcd(int a,int b,int &x,int &y)
{
	if(!b)
	{
		x=1,y=0;
		return a;
	}
	int d=exgcd(b,a%b,y,x);
	y-=a/b*x;
	return d;
}
bool check(int m)
{
	for(int i=1;i<=n;++i)
	for(int j=i+1;j<=n;++j)
	{
		int a=p[i]-p[j],b=m,c=s[j]-s[i],x,y;
		int d=exgcd(a,b,x,y);
		if(c%d) continue;
		a/=d,b/=d,c/=d;
		if(b<0) b=-b;
		x=(x*c%b+b)%b;
		if(x<=l[i]&&x<=l[j]) return 0; 
	}
	return 1;
}
int main()
{
	scanf("%d",&n);
	int mx=0;
	for(int i=1;i<=n;++i)
	scanf("%d%d%d",&s[i],&p[i],&l[i]),mx=std::max(mx,s[i]);
	for(int i=mx;;++i)
	if(check(i))
	return printf("%d",i),0;
	return 0;
}

上一题