洛谷 P2312 & bzoj 3751 解方程 —— 取模

题目:https://www.luogu.org/problemnew/show/P2312

https://www.lydsy.com/JudgeOnline/problem.php?id=3751

10^10000 太大了,高精度也很难做,怎么办?

注意我们要求的是方程的值 = 0 的解,不妨在取模意义下做,因为真正使方程 = 0 的解在模意义下也是 0;

然后可以用秦九韶算法,O(n) 算每个枚举的答案;

避免出错要多对几个数取模,就像哈希时有多个模数一样;

据说模数大小在 2e4 左右比较好;

模数多了会 T,少了会错,最后取了5个才勉强过去;

考试时遇到这种题怎么估计...

代码如下:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
int const maxn=105,maxm=1e6+5;
int n,m,a[10][maxn],p[10]={0,29501,18919,15671,30271,24443};
int pri[maxm],cnt,ans[maxm],c=5;
bool vis[maxm],fl[10][200005];
int main()
{
  scanf("%d%d",&n,&m);
  for(int i=0;i<=n;i++)
    {
      int f=1; char ch=getchar();
      while(ch<'0'||ch>'9'){if(ch=='-')f=-1; ch=getchar();}
      while(ch>='0'&&ch<='9')
    {
      for(int k=1;k<=c;k++)
        a[k][i]=(a[k][i]*10+ch-'0')%p[k];
      ch=getchar();
    }
      for(int k=1;k<=c;k++)a[k][i]=(a[k][i]*f+p[k])%p[k];//!
    }
  int cnt=0;
  for(int k=1;k<=c;k++)
    for(int x=1;x<p[k]&&x<=m;x++)//
    {
      int ret=0;
      for(int i=n;i>=0;i--)ret=((ll)ret*x%p[k]+a[k][i])%p[k];
      if(ret)fl[k][x]=1;
    }
  for(int i=1;i<=m;i++)
    {
      bool flag=0;
      for(int k=1;k<=c;k++)if(fl[k][i%p[k]]){flag=1; break;};
      if(!flag)ans[++cnt]=i;
    }
  printf("%d\n",cnt);
  for(int i=1;i<=cnt;i++)printf("%d\n",ans[i]);
  return 0;
}

 

posted @ 2018-09-13 21:42  Zinn  阅读(606)  评论(0编辑  收藏  举报