bzoj1407 / P2421 [NOI2002]荒岛野人(exgcd)

P2421 [NOI2002]荒岛野人

洞穴数不超过1e6 ---> 枚举

判断每个野人两两之间是否发生冲突:exgcd

假设有$m$个洞穴,某两人(设为1,2)在$t$时刻发生冲突

那么我们可以列出方程

$c_{1}+p_{1}tequiv c_{2}+p_{2}t (modquad m)$

移项一下:$(p_{1}-p_{2})tequiv c_{2}-c_{1} (modquad m)$

去掉$(mod m)$,得$(p_{1}-p_{2})t-mx=c_{2}-c_{1} $

这不就是不定方程标准形式$ax+by=c$吗!

用exgcd搞搞,求出$t$的最小正整数解

如果$t<=min(l_{1},l_{2})$说明在两只的有生之年会发生冲突

此时$m$是不合法的,向下枚举就行了

复杂度$O(1e6n^{2}logp_{max})$,然鹅实际效率远高于介个

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #define re register
 5 using namespace std;
 6 int min(int a,int b){return a<b?a:b;}
 7 int max(int a,int b){return a>b?a:b;}
 8 #define N 17
 9 int n,C[N],p[N],l[N],x0,y0,g;
10 void exgcd(int a,int b,int &x,int &y){
11     if(!b) g=a,x=1,y=0;
12     else exgcd(b,a%b,y,x),y-=x*(a/b);
13 }
14 bool check(int b){
15     for(int i=1,a,c,q;i<=n;++i)
16         for(int j=i+1;j<=n;++j){
17             a=((p[i]-p[j])%b+b)%b,c=C[j]-C[i];
18             exgcd(a,b,x0,y0);q=b/g;//求ax+by=gcd(a,b)
19             if(c%g) continue;
20             x0=(x0*c/g%q+q)%q;//先*c/g转成原来的式子,再%(b/gcd(a,b))得到最小解
21             if(0<=x0&&x0<=min(l[i],l[j]))
22                 return 0;
23         }
24     return 1;
25 }
26 int main(){
27     scanf("%d",&n); int st=0;
28     for(int i=1;i<=n;++i)
29         scanf("%d%d%d",&C[i],&p[i],&l[i]),st=max(st,C[i]);
30     for(re int i=st;i<=1000000;++i)//注意是从洞穴编号最大的那个开始枚举
31         if(check(i)){printf("%d",i);break;}
32     return 0;
33 }
View Code
原文地址:https://www.cnblogs.com/kafuuchino/p/9899152.html