Reducing Fractions CodeForces - 222C

原题链接
考察:质因数分解,筛质数
思路:
  比较明显的思路是循环\(1\)~\(n\)每一个\(a[i]\),分解质因数,时间复杂度约为n*\(\sqrt{10^7}\)会有超时的风险.
  优化就是筛到\(10^7\)的质数,然后判断为质数立马跳出.
  将质数的指数取最小值.然后循环\(a[i],b[i]\)除去质数.
  但是这里不适合指数取原值\(-\)最小值,因为求积时会爆long long.

Code

#include <iostream>
#include <cstring>
#include <vector>
using namespace std;
typedef long long LL;
const int N = 1e7+10,M = 100010;
int n,m,prime[N],cnt,a[M],b[M];
int pa[N],pb[N];
bool st[N];
void GetPrime(int n)
{
	for(int i=2;i<=n;i++)
	{
		if(!st[i]) prime[++cnt] = i;
		for(int j=1;prime[j]<=n/i;j++)
		{
			st[i*prime[j]] = 1;
			if(i%prime[j]==0) break;
		}
	}
}
void div(int s,int ps[])
{
	for(int i=1;prime[i]<=s/prime[i]&&st[s];i++)
	{
		if(s%prime[i]==0)
		{
			int sz = 0;
			while(s%prime[i]==0) s/=prime[i],ps[prime[i]]++;
		}
	}
	if(s>1) ps[s]++;
}
LL qsm(int a,int k)
{
	LL res = 1;
	while(k)
	{
		if(k&1) res = (LL)a*res;
		a = (LL)a*a;
		k>>=1;
	}
	return res;
}
void reduce(int& x,int ps[])
{
	int res = x;
	for(int i=1;prime[i]<=x/prime[i];i++)
	{
		if(!st[x]) break;
		if(x%prime[i]==0)
		{
			while(x%prime[i]==0)
			{
				x/=prime[i];
				if(ps[prime[i]]) ps[prime[i]]--,res/=prime[i];
			}
		}
	}
	if(x>1&&ps[x]) ps[x]--,res/=x,x = 1;
	x = res;
}
int main()
{
	scanf("%d%d",&n,&m);
	GetPrime(N-1);
	for(int i=1;i<=n;i++)
	{
		scanf("%d",&a[i]);
		div(a[i],pa);
	}
	for(int i=1;i<=m;i++)
	{
		scanf("%d",&b[i]);
		div(b[i],pb);
	}
	for(int i=1;i<=cnt;i++)
	{
		int ca = pa[prime[i]],cb = pb[prime[i]];
		pa[prime[i]]=min(ca,cb);
		pb[prime[i]]=min(ca,cb);
	}
	for(int i=1;i<=n;i++) reduce(a[i],pa);
	for(int i=1;i<=m;i++) reduce(b[i],pb);
	printf("%d %d\n",n,m);
	for(int i=1;i<=n;i++) printf("%d ",a[i]);
	printf("\n");
	for(int i=1;i<=m;i++) printf("%d ",b[i]);
	printf("\n");
	return 0;
}
posted @ 2021-08-31 22:42  acmloser  阅读(53)  评论(0编辑  收藏  举报