2023.10.12

大抵是没有挂分。

简单题+博弈+图论+树论,典。


xor

一个 \(n\times n\) 的空矩阵 \(A\),进行如下操作:

给定 \(r,c,l,s\),对于 \(x\in[r,r+l)\)\(y\in[c,x-r+c]\),给 \(A_{x,y}\) 加上 \(s\),也就是以 \((r,c)\) 为左上角的下三角区域。

最后问整个矩形的异或和。

\(n\le 1000\)\(q\le 3\times 10^5\)\(s\le 10^9\).


肯定是先求出每个位置的值。

容易想到每行差分,所以维护差分的列差分和斜线差分即可。

点击查看代码
#include<bits/stdc++.h>
#define ll long long
#define N 1010
#define pii pair<int,int>
#define mp make_pair
#define fi first
#define se second
using namespace std;
int read(){
	int x=0,w=1;char ch=getchar();
	while(!isdigit(ch)){if(ch=='-')w=-1;ch=getchar();}
	while(isdigit(ch))x=(x<<3)+(x<<1)+(ch^48),ch=getchar();
	return x*w;
}
int n,q,dlt;
ll a[N][N];
ll b[N][N],c[N<<1][N];
pii getpos(int i,int j){
	i-=dlt;
	if(i==0)return mp(j,j);
	if(i<0)return mp(1+(j-(1-i)),j);
	return mp((i+1)+(j-1),j);
}
bool check(pii x){
	return x.fi>0&&x.fi<=n&&x.se>0&&x.se<=n;
}
int main(){
	freopen("xor.in","r",stdin);
	freopen("xor.out","w",stdout);
	n=dlt=read(),q=read();
	for(int x,y,l,s;q;q--){
		x=read(),y=read(),l=read(),s=read();
		b[x][y]+=s,b[min(n+1,x+l)][y]-=s;
		c[x-(y+1)+dlt][y+1]-=s,c[x-(y+1)+dlt][min(n+1,y+l+1)]+=s;
	}
	pii tp;
	for(int i=0;i<=dlt*2;i++)
		for(int j=1;j<=n;j++){
			c[i][j]+=c[i][j-1];
			tp=getpos(i,j);
			if(check(tp))a[tp.fi][tp.se]=c[i][j];
		}
	for(int j=1;j<=n;j++)
		for(int i=1;i<=n;i++)
			b[i][j]+=b[i-1][j],a[i][j]+=b[i][j];
	ll ans=0;
	for(int i=1;i<=n;i++)
		for(int j=1;j<=n;j++)
			a[i][j]+=a[i][j-1],ans^=a[i][j];
	printf("%lld\n",ans);
	
	return 0;
}

game

两人博弈。初始有一个集合 \(S=\{a_n\}\).

\(i\in[1,m]\) 轮 有一个参数 \(b_i\),操作者仅能只留下或全部舍弃 \(S\) 中所有 \(b_i\) 的倍数。

令操作结束后 \(S\)(可空)中的元素和为 \(x\),先手欲最小化 \(x\),后手欲最大化 \(x\).

问最后的 \(x\).

\(n\le 2\times 10^4\)\(m\le 2\times 10^5\)\(|a_i|\le 4\times 10^{14}\)\(1\le b_i\le 4\times 10^{14}\).


部分分刷爽的一集。没什么必要写就不写了。

结论

对于先手,若每次选择集合大小更小的一侧,\(\log n\) 次可使答案为 \(0\),后手同理。

也就是说当 \(m> 2\log n\) 时答案必为 \(0\),接下来考虑一个有时间复杂度保证的算法。

使用二叉树构建整个博弈过程。

  • insert

若当前节点不存在,创建一个新节点。当其为 \(b_{dep}\) 的倍数时,插入右子树,反之插入左子树。

  • query

查询当前轮数下的最大/最小权值。在左右子树中查询,并按照先后手决策计算答案。

每次决策会将集合划分为两个不交集合,故时间复杂度 \(O(nm)\).

总时间复杂度 \(O(n\log n)\).

点击查看代码
#include<bits/stdc++.h>
#define ll long long
#define N 20010
#define M 200010
#define L 30
using namespace std;
ll read(){
	ll x=0,w=1;char ch=getchar();
	while(!isdigit(ch)){if(ch=='-')w=-1;ch=getchar();}
	while(isdigit(ch))x=(x<<3)+(x<<1)+(ch^48),ch=getchar();
	return x*w;
}
int n,m;
ll a[N],b[M];
struct Tree{
	ll s[N*L];
	int lc[N*L],rc[N*L],tot;
	void insert(int &p,int dep,ll val){
		if(!p)p=++tot;
		if(dep>m)return s[p]+=val,void();
		if(val%b[dep])insert(lc[p],dep+1,val);
		else insert(rc[p],dep+1,val);
	}
	ll query(int p,int dep){
		if(!p)return 0;
		if(dep>m)return s[p];
		ll r1=query(lc[p],dep+1),r2=query(rc[p],dep+1);
		return (dep&1)?min(r1,r2):max(r1,r2);
	}
}T;
int rt;
int main(){
	freopen("game.in","r",stdin);
	freopen("game.out","w",stdout);
	n=read(),m=read();
	for(int i=1;i<=n;i++)a[i]=read();
	for(int i=1;i<=m;i++)b[i]=read();
	if(m>28)return puts("0"),0;
	for(int i=1;i<=n;i++)
		T.insert(rt,1,a[i]);
	printf("%lld\n",T.query(rt,1));
	
	return 0;
}

connect

无边的图,对于 \(1\le i<j\le n\),若 \(\gcd(i,j)\) 为合数则在 \(i,j\) 间连无向边。

试删除一个节点,最小化最大的连通块大小。多测。

\(T\le 10\)\(n\le 10^5\)\(a_i\le 10^7\).


有意义的删点显然在割点上。对于较大的 \(n\) 考虑优化建边。

令单位合数为可以表示为 \(2\) 个质数的乘积的数。

对于存在边的两点,将它们与 \(\gcd\) 的一个是单位合数的因子连边。

每个数的单位合数因子是 \(O(\log V^2)\) 的,也就是新图总边数 \(O(n\log V^2)\).

直接在原图最大的连通块里 tarjan.

遇到返祖边时,可以在栈里不断跳回去,把 \(siz\) 加回 \(u\) 里。

不训图论导致的。

点击查看代码
#include<bits/stdc++.h>
#define N 100010
#define M 2005010
#define E 6000010
#define R 10000001
using namespace std;
int read(){
	int x=0,w=1;char ch=getchar();
	while(!isdigit(ch)){if(ch=='-')w=-1;ch=getchar();}
	while(isdigit(ch))x=(x<<3)+(x<<1)+(ch^48),ch=getchar();
	return x*w;
}
int p[R>>3],minp[R],id[R],cnt,P;bool vis[R];
int fa[M],sz[M],siz[M];
int find(int x){
	return fa[x]==x?x:fa[x]=find(fa[x]);
}
void merge(int u,int v){
	u=find(u),v=find(v);
	if(u==v)return;
	fa[u]=v,sz[v]+=sz[u];
}
int dfn[M],low[M],st[M],top,tim;
void init(){
	for(int i=2;i<R;i++){
		if(!vis[i])p[++cnt]=minp[i]=i;
		for(int j=1;j<=cnt&&i*p[j]<R;j++){
			vis[i*p[j]]=true,minp[i*p[j]]=p[j];
			if(i%p[j]==0)break;
		}
	}
	for(int i=2;i<R;i++)
		if(vis[i]&&!vis[i/minp[i]])id[i]=++P;
}
int head[M],nxt[E],ver[E],tot;
void add(int u,int v){
	nxt[++tot]=head[u];
	ver[tot]=v;
	head[u]=tot;
}
int ans,node;
int n;
void tarjan(int u,int edge){
	dfn[u]=low[u]=++tim;
	st[++top]=u,siz[u]=0;
	int mx=0,scc=0;
	for(int i=head[u],v;i;i=nxt[i]){
		if(i==(edge^1))continue;
		if(!dfn[v=ver[i]]){
			tarjan(v,i),low[u]=min(low[u],low[v]);
			if(low[v]>=dfn[u]){
				int sum=0;
				for(int x=0;x!=v;)
					x=st[top--],siz[u]+=siz[x],sum+=siz[x];
				scc++,mx=max(mx,sum);
			}
		}
		else low[u]=min(low[u],dfn[v]);
	}
	if(u<=n){
		siz[u]++;
		if((edge&&scc)||(!edge&&scc>1))
				ans=min(ans,max(mx,node-siz[u]));
		else ans=min(ans,node-1);
	}
}
void solve(){
	memset(head,0,sizeof(head)),tot=1,tim=top=0;
	n=read();
	for(int i=1;i<=n+P;i++)
		fa[i]=i,sz[i]=dfn[i]=low[i]=0;
	int pr[10],c[10];
	for(int i=1,x,t;i<=n;i++){
		x=read(),sz[i]=1;
		for(int j=0;j<10;j++)pr[j]=c[j]=0;
		while(x>1){
			t=minp[x],pr[++pr[0]]=t;
			while(x%t==0)x/=t,c[pr[0]]++;
		}
		for(int j=1;j<=pr[0];j++)
			for(int k=j+(c[j]<=1);k<=pr[0];k++)
				if(1ll*pr[j]*pr[k]<R){
					int tp=id[pr[j]*pr[k]]+n;
					add(i,tp),add(tp,i);
					merge(tp,i);
				}
	}
	int mx=0,se=0,pos=0;
	for(int i=1;i<=n+P;i++){
		if(fa[i]!=i||!sz[i])continue;
		if(sz[i]>mx)se=mx,mx=sz[i],pos=i;
		else if(sz[i]>se)se=sz[i];
	}
	ans=node=mx;
	tarjan(pos,0);
	printf("%d\n",max(ans,se));
}
int main(){
	freopen("connect.in","r",stdin);
	freopen("connect.out","w",stdout);
	init();
	int T=read();
	while(T--)solve();
	
	return 0;
}

route

U161443 [NOI2021SDPTTest5]平凡

留给想做的同学。


100 + 60 + 15 + 0 = 175.

posted @ 2023-10-12 20:23  SError  阅读(13)  评论(0编辑  收藏  举报