国庆day3总结--上午
Crash_Man_CHN · · 个人记录
1.求质数
(1)埃氏筛
时间复杂度:O(n*loglogn)
大致思路:用标记数组标记它平方以上的倍数,若最后还没有标记的那就是质数
模板代码
void e_prime()
{
vis[1]=1;
for(int i=2;i*i<=n;i++)
{
if(vis[i]==0)
{
for(int j=i*i;j<=n;j+=i)vis[j]=1;
}
}
}
例题
1.P3912 素数个数
模板题,但是要注意数组大小和vis要是bool
#include <bits/stdc++.h> using namespace std; const int N=1e8+5; int n,ans; bool vis[N]; void e_prime() { vis[1]=1; for(int i=2;i*i<=n;i++) { if(vis[i]==0) { for(int j=i*i;j<=n;j+=i) { vis[j]=1; } } } } int main() { cin>>n; e_prime(); for(int i=2;i<=n;i++) { if(vis[i]==0)ans++; } cout<<ans; return 0; }2.P1621 集合
模板+并查集,主要是考的对埃氏筛的理解
主要思路
在求质数的同时判断p,并且合并,最后判断集合即可
代码
#include <bits/stdc++.h>
#define int long long
using namespace std;
const int N=1e5+5;
int a,b,p,fa[N],cnt;
bool vis[N];
int find(int x)
{
if(fa[x]==x)return x;
return fa[x]=find(fa[x]);
}
void unionn(int x,int y)
{
x=find(x),y=find(y);
if(x==y)return ;
fa[x]=y;
}
void e_prime()
{
vis[1]=1;
for(int i=2;i<=b;i++)
{
if(vis[i]==0)
{
for(int j=i*2;j<=b;j+=i)
{
vis[j]=1;
if(i>=p)unionn(i,j);
}
}
}
}
void solve()
{
for(int i=1;i<=b;i++)fa[i]=i;
e_prime();
int res=0;
for(int i=a;i<=b;i++)
{
if(fa[i]==i)res++;
}
cout<<res;
return ;
}
signed main()
{
ios::sync_with_stdio(false);
cin.tie(0),cout.tie(0);
cin>>a>>b>>p;
solve();
return 0;
}
(2)空间时间都高的质数题目
当一个题目空间时间都高的时候,该怎么办呢? 不妨将他砍半,比如2^32变为2^16,也就是65536
例题
1.UVA10140/P1835 素数密度
主要思路
1.无法筛出1~R的所有质数,时间TLE,标记数组空间MLE
2.因为R-L不超过1e6,那么L~R是可以枚举的
3.任意一个荷属x只要被他最小的质因数p标记即可 -> 这时p<=sqrt(x)
4.只要将2^16以内(65536)的质数筛出来,就足以标记2^32以内的所有合数
5.空间问题可以用平移即可解决->vis[i-L]=1而不是vis[i]=1
6.当标记L~R的所有合数后再枚举L~R维护ans表示上一个找到的质数即可
实现代码
#include <bits/stdc++.h>
#define int long long
using namespace std;
const int N=1e6+5;
int l,r,x,y,tot;
bool vis[N];
int prime[N];
void e_prime(int n)
{
vis[1]=1;
for(int i=2;i*i<=n;i++)
{
if(vis[i]==0)
{
for(int j=i*i;j<=n;j+=i)
{
vis[j]=1;
}
}
}
for(int i=2;i<=65536;i++)
{
if(!vis[i])prime[++tot]=i;
}
}
signed main()
{
ios::sync_with_stdio(false);
cin.tie(0),cout.tie(0);
e_prime(65536);
memset(vis,0,sizeof(vis));
cin>>l>>r;
for(int i=1;i<=tot;i++)
{
int p=prime[i],x=ceil(l*1.0/p);
for(int j=max(2ll,x)*p;j<=r;j+=p)vis[j-l]=1;
}
int cnt=0;
int maxx=-1e18,minn=1e18,ans=0,mxa=0,mxb=0,mia=0,mib=0;
for(int i=l;i<=r;i++)
{
if(!vis[i-l])cnt++;
}
if(l==1)cnt--;
cout<<cnt<<"\n";
return 0;
}
因数分解
简单的质因数分解
(1)不需要思考直接按题意模拟
例题
1.B3871 [GESP202309 五级] 因数分解
模板题
#include <bits/stdc++.h> #define int long long using namespace std; const int N=1e6+5; bool is_prime(int n) { if(n<2)return 0; for(int i=2;i*i<=n;i++) { if(n%i==0)return 0; } return 1; } signed main() { ios::sync_with_stdio(false); cin.tie(0),cout.tie(0); int n; cin>>n; if(is_prime(n)) { cout<<n; return 0; } for(int i=2;i*i<=n;i++) { int cnt=0;//统计次数 while(n%i==0)n/=i,cnt++; if(cnt==0)continue; else if(cnt>1) { cout<<i<<"^"<<cnt; if(n==0)break; else cout<<" * "; } else { cout<<i; if(n==0)break; else cout<<" * "; } } if(n)cout<<n; return 0; }
(2)技巧性题目
例题
1.P1469 找筷子
实现思路
直接异或
#include <bits/stdc++.h> #define int long long using namespace std;
signed main() { ios::sync_with_stdio(false); cin.tie(0),cout.tie(0); int n,ans=0; cin>>n; for(int i=1;i<=n;i++) { int x; cin>>x; ans^=x; } cout<<ans; return 0; }