2021“MINIEYE杯”中国大学生算法设计超级联赛1

1001 Mod, Or and Everything

题目大意
\(\sum_{i=1}^{n-1}n\ mod\ i\)
\(n<=10^{12}\)
打表找规律n的答案是小于n的第一个\(2^x\)再减去1。

#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<algorithm>
using namespace std;
#define int long long
int read(){
    int sum=0,f=1;
    char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
    return sum*f;
}
int work(int x){
    int num=0;
    x>>=1;
    while(x){
        x>>=1;
        num=num<<1|1;
    }
    return num;
}
signed main(){
    int T=read();
    while(T--){
        int ans=0;
        int n=read();
        cout<<work(n-1)<<endl; 
    }
    return 0; 
}

1005

求2~n组成边权为\(lcm(u[i],v[i])\)生成树的边权之和的最小值。
为使答案最小,质数连2,合数连一个约数。统计答案即可。

#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<algorithm>
using namespace std;
#define int long long
#define MAXN 1e7
#define N 10000000
int read(){
    int sum=0,f=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
    return sum*f;
}
int vis[N],phi[N],pri[N],cnt,ans[N]; 
void init() {
  phi[1] = 1;//0是质数 
  for (int i = 2; i < MAXN; ++i) {
    if (!vis[i]) {
      phi[i] = i - 1;
      pri[cnt++] = i;
    }
    for (int j = 0; j < cnt; ++j) {
      if (1ll * i * pri[j] >= MAXN) break;
      vis[i * pri[j]] = 1;
      if (i % pri[j]) {
        phi[i * pri[j]] = phi[i] * (pri[j] - 1);
      } else {
        phi[i * pri[j]] = phi[i] * pri[j];
        break;
      }
    }
  }
}
void pre_work(){
    ans[2]=0;
    for(int i=3;i<=N;i++){
        if(vis[i]==0)ans[i]=ans[i-1]+i*2;
        else ans[i]=ans[i-1]+i;
    }
}
signed main(){
    int T=read();
    init();
    pre_work();
    while(T--){
        int n=read();
        cout<<ans[n]<<endl;
    }
}

1006

给一个数列,求一个最短的区间使其区间异或和大于k
\(n<=100000\)
01trie上维护子树中出现过的前缀异或和在数列中最大位置。然后对于每一个r判断从1~r找到最优答案。

#include<iostream>
#include<cstring>
#include<cmath>
#include<cstdio>
#include<algorithm>
using namespace std;
#define INF 2147483647
#define N 500100
int ch[N*50][2],mx_pos[N*50],root,tot,sum[N];
int read(){
	int sum=0,f=1;char ch=getchar();
	while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
	while(ch>='0'&&ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
	return sum*f;
}
void update(int now){
	if(ch[now][0])mx_pos[now]=max(mx_pos[ch[now][0]],mx_pos[now]);
	if(ch[now][1])mx_pos[now]=max(mx_pos[ch[now][1]],mx_pos[now]);
}
void add(int tall,int pos,int x,int &now){
	if(now==0)now=++tot;
	if(tall==0){mx_pos[now]=pos;return;}
	int bit=x>>tall-1&1;
	add(tall-1,pos,x,ch[now][bit]); 
	update(now);
}
int check(int tall,int x,int k,int now){
	int bit_k=k>>tall-1&1;
	int bit_x=x>>tall-1&1;
	if(tall==0)return mx_pos[now];
	if(bit_k==0)return max(check(tall-1,x,k,ch[now][bit_x]),mx_pos[ch[now][bit_x^1]]);
	else return mx_pos[ch[now][bit_x^1]]==-1?-1:check(tall-1,x,k,ch[now][bit_x^1]);
}
void init(){
	while(tot){
		mx_pos[tot]=-1;
		ch[tot][0]=ch[tot][1]=0;
		tot--;
	}
	root=0;
}
int main(){
	int T=read();
	memset(mx_pos,-1,sizeof(mx_pos));
	while(T--){
		init(); 
		int ans=INF,ansl=INF,ansr=INF;
		int n=read(),k=read();
		add(31,0,0,root);
		for(int i=1;i<=n;i++){
			sum[i]=sum[i-1]^read();
			int l=check(31,sum[i],k,root)+1;
			add(31,i,sum[i],root);
			if(l==0)continue;
			int tmp=i-l+1;
			if(tmp<ans)ans=tmp,ansl=l,ansr=i;
		}
		if(ans==INF)printf("-1\n");
		else printf("%d %d\n",ansl,ansr);
	}
	return 0;
}

1007

打表然后发现\(f[i]=f[i-1]*(n-1)+(-1)^x(n-1)\)
推一推式子得到\(\frac{n-1}{n}[(n-1)^{t-1}+(-1)^t]=x\)
然后对于t的奇偶讨论,用BSGS求答案就好。

#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<algorithm>
#include<unordered_map>
using namespace std;
#define N 40000
#define int long long
unordered_map<int,int> Map1,Map2;
int p,m;
int ksm(int x,int b){
	int ans=1;
	while(b){
		if(b&1)ans=ans*x%p;
		b>>=1;
		x=x*x%p;
	}
	return ans;
}
int BSGS(int n,int x1,int x2){
	int tmp=1;
	for(int i=1;i<=m;i++){
		tmp=tmp*n%p;
		Map1[tmp*x1%p]=i;
		Map2[tmp*x2%p]=i;
	}
	n=ksm(n,m);tmp=1;
	for(int i=1;i<=m+1;i++){
		tmp=tmp*n%p;
		int w1=Map1.count(tmp);
		int w2=Map2.count(tmp);
		if(w2)return (i*m-Map2[tmp]+p)*2%p;
		if(w1)return ((i*m-Map1[tmp]+p)*2+1)%p;
	}
	return -1;
} 
int read(){
	int sum=0,f=1;char ch=getchar();
	while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
	while(ch>='0'&&ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
	return sum*f;
}
signed main(){
	p=998244353;m=sqrt(p)+1;
	int T=read();
	while(T--){
		int n=read(),x=read();
		if(x==1){
			printf("0\n");
			continue;
		}
		if(x==0){
			printf("1\n");
			continue;
		}
		x=x*n%p*ksm(n-1,p-2)%p;
		int base=(n-1)*(n-1)%p; 
		Map1.clear();
		Map2.clear();
		int f1=(x+1)%p;
		int f2=(x-1+p)*(n-1)%p;
		int ans=BSGS(base,f1,f2);
		printf("%lld\n",ans);
	}
	return 0;
}

1008

洛谷原题,《月蟾宫》

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
int n,m,s[2010],l[2010],ans,a[2010][2010],num[2010][2010];
int main(){
    int T;
    scanf("%d",&T);
    while(T--){
        int n,m;
        ans=0;
        memset(a,0,sizeof(a));
        memset(num,0,sizeof(num));
        memset(s,0,sizeof(s));
        memset(l,0,sizeof(l));
        scanf("%d %d",&n,&m);
        for(int i=1;i<=n;++i)
            for(int j=1;j<=m;++j){
                scanf("%d",&num[i][j]);
                if(num[i][j]>=num[i-1][j]) a[i][j]=a[i-1][j]+1;
                else a[i][j]=1;
            }
        for(int top,len,i=1;i<=n;++i){
            top=0; len=0; s[top]=0; l[top]=0;
            for(int j=1;j<=m;++j){
                if(a[i][j]>=s[top]){
                    s[++top]=a[i][j];
                    l[top]=1;
                }else{
                    len=0;
                    while(top&&s[top]>a[i][j]){
                        len+=l[top];
                        ans=max(ans,len*s[top]);
                        --top;
                    }
                    s[++top]=a[i][j];
                    l[top]=len+1;
                }
            }
            len=0;
            while(top){
                len+=l[top];
                ans=max(ans,len*s[top]);
                --top;
               }
           }
           printf("%d\n",ans);
    }
    return 0;
}

1009

二分权值用并查集维护连通块即可

#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<algorithm>
using namespace std;
#define N 101000
#define M 501000
int fa[N],x[M],w[M],y[M],col[N],n,m,k;
int find(int x){
    if(fa[x]==x)return x;
    else return fa[x]=find(fa[x]);
}
int check(int W){
    for(int i=1;i<=n;i++)fa[i]=i;
    for(int i=1;i<=m;i++){
        if(w[i]<=W){
            int fx=find(x[i]);
            int fy=find(y[i]);
            fa[fx]=fy;
        }
    }
    for(int i=1;i<=n;i++)col[i]=0;
    for(int i=1;i<=n;i++)col[find(i)]=1;
    int ans=0;
    for(int i=1;i<=n;i++)ans+=col[i];
    return ans;
}
int read(){
    int sum=0,f=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
    return sum*f;
}
int main(){
    int T=read();
    while(T--){
        n=read(),m=read(),k=read();
        for(int i=1;i<=m;i++)x[i]=read(),y[i]=read(),w[i]=read();
        int l=0,r=1000000001;
        int ans=-1;
        while(l<=r){
            int mid=(l+r)>>1;
            int tmp=check(mid);
            if(tmp<=k){
                if(tmp==k)ans=mid;
                r=mid-1;
            }
            else l=mid+1;
        }
        printf("%d\n",ans);
    }
    return 0;
}

1010

HH项链原题,就是询问一个区间的某个值域有多少不同的数。
离线然后树状数组套权值线段树。

#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<algorithm>
using namespace std;
#define N 101000
#define mid (L+R>>1)
int sum[N*400],ch[N*400][2],tot,n,m,root[N*20];
int a[N],pre[N],pos[N],ans[N];
struct qu{
    int x0,x1,y0,y1,num;
}q[N];
int read(){
    int sum=0,f=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
    return sum*f;
}
void update(int now){
    sum[now]=sum[ch[now][0]]+sum[ch[now][1]];
}
void add(int L,int R,int x,int w,int &now){
    if(now==0)now=++tot;
    if(L==R){
        sum[now]+=w;
        return;
    }
    if(x>mid)add(mid+1,R,x,w,ch[now][1]);
    else add(L,mid,x,w,ch[now][0]);
    update(now);
}
int check(int L,int R,int l,int r,int now){
    if(now==0)return 0;
    if(L==l&&R==r)return sum[now];
    if(l>mid)return check(mid+1,R,l,r,ch[now][1]);
    else if(r<=mid)return check(L,mid,l,r,ch[now][0]);
    else  return check(L,mid,l,mid,ch[now][0])+check(mid+1,R,mid+1,r,ch[now][1]);
}
bool cmp(qu a,qu b){
    return a.x1<b.x1;
}
int lowbit(int x){
    return x&-x;
}
void ADD(int x,int val,int w){
    for(int i=x;i<=n;i+=lowbit(i)){
        add(1,100001,val,w,root[i]);
    }
}
int query(int x,int l,int r){
    int ans=0;
    for(int i=x;i>=1;i-=lowbit(i)){
        ans+=check(1,100001,l,r,root[i]);
    }
    return ans;
}
int main(){
    int T=read();
    while(T--){
        while(tot){
            ch[tot][0]=ch[tot][1]=sum[tot]=0;
            tot--;
        }
        n=m=0;
        memset(ans,0,sizeof(ans));
        memset(pos,0,sizeof(ans));
        memset(pre,0,sizeof(ans));
        memset(a,0,sizeof(ans));
        memset(root,0,sizeof(ans));
        n=read(),m=read();
        for(int i=1;i<=n;i++){
            a[i]=read()+1;
            pre[i]=pos[a[i]]+1;
            pos[a[i]]=i;
        }
        for(int j=1;j<=m;j++)q[j].x0=read(),q[j].y0=read()+1,q[j].x1=read(),q[j].y1=read()+1,q[j].num=j;
        sort(q+1,q+1+m,cmp);
        for(int now=0,i=1;i<=m;i++){
            while(now<q[i].x1){
                now++;
                ADD(pre[now],a[now],+1);
                ADD(now+1,a[now],-1);
            }
            ans[q[i].num]=query(q[i].x0,q[i].y0,q[i].y1);
        }
        for(int i=1;i<=m;i++)
            printf("%d\n",ans[i]);
    }
    return 0; 
}