2024.7.23 模拟赛6

模拟赛

T1 就是 \(\mathbb{A}\) 不了!!!

T1 mod M

唐了一个半小时,最后 40min 才看出来,莫名挂 \(6\) 分。

如果只考虑 \(mod 2\) 的情况。最终答案最多有两种。

那显然,我们只需要考虑什么时候能剩下一种。

只有在 \(n\) 个数都同余时才能剩下一种。

既然都同余,就是任意两数的差的 \(gcd\) 都不为 \(1\)

思维不够严密,有 \(n=2\) 或其他卡边界的状态会寄,以下方法可以避免。

code
#include<bits/stdc++.h>
using namespace std;
const int N = 2e5+5;
int n,a[N];

int main()
{
	scanf("%d",&n); 
	for(int i=1;i<=n;i++) scanf("%d",&a[i]);
	sort(a+1,a+1+n);
	for(int i=2;i<=n;i++)
	{
		if(a[i]==a[i+1]) continue;
		if(__gcd(a[i]-a[1],a[i]-a[i-1])==1) return printf("2\n"),0;
	}
	printf("1\n");
	return 0;
}

T2 Number of Multisets

其实挺简单的一道 dp,赛时唐。

想到去年 5K 口胡的题:

  • \(k\) 个正整数相加等于 \(m\),求方案数。

正解类似分讨,\(f_{i,j}\) 表示选 \(i\) 个数和为 \(j\) 的方案数:

  • 考虑包含 \(1\) 的方案,将 \(1\) 删掉,那么这类方案的数量为 \(f_{i-1,j-1}\)

  • 考虑剩下的方案一定都不包含 \(1\),那么把 \(i\) 个数都减去 \(1\)。方案数为 \(f_{i,j-i}\)

回来看这道题,我们仍是把所有方案划分成两类:包含 \(1\),不包含 \(1\)

那么所有包含 \(1\) 的方案可以由 \(f_{i-1,j-1}\) 得到,不包含 \(1\) 的可以把每个数乘二,由 \(f_{i,j\ times 2}\) 得到。

code
#include<bits/stdc++.h>
using namespace std;
const int N = 3e3+5,mod = 998244353;
int n,k;
int f[N][N];
int main()
{
	scanf("%d%d",&n,&k);
	f[0][0]=1;
	for(int i=1;i<=n;i++) 
	{
		f[i][i]=1;
		for(int j=i;j>=1;j--)
		{
			f[i][j]=(f[i-1][j-1]+f[i][j<<1])%mod;
		}
	}
	printf("%d\n",f[n][k]);
	return 0;
}

T3 Simultaneous Sugoroku

首先拿暴力和值域的部分分,都好拿。值域的其实有一点启发正解。

移动机器人其实和移动原点是等价的,假如我们把每次操作看成对原点移动,那么会有一个显然的性质:

  • 关于原点对称的两个点在之后的操作中位置一定是对称的。

举例:假如原点移动到了 \(x\),那么 \(x-i\)\(x+i\) 最终答案一定为相反数。

因此每次都有一半的数轴可以直接对称过去,也就是每次能删掉一部分。

我们可以用带权并查集维护对称次数,并查集的范围就是值域范围。

维护一个区间表示数轴上的实际范围,另一个维护以当前原点为中心左右延申的距离,每次合并较小的一半(启发式?)。

code
#include<bits/stdc++.h>
using namespace std;
const int N = 3e5+5,M = 1e6+6;
int n,m,a[N],b[N],ans[M];
int l,r,mid,L,R;
int fa[M];bool tag[M];
int find(int x)
{
	if(fa[x]!=x) 
	{
		int fx=fa[x];
		fa[x]=find(fa[x]);
		tag[x]^=tag[fx];
	}
	return fa[x];
}
int main()
{
	scanf("%d%d",&n,&m);
	for(int i=1;i<=n;i++) scanf("%d",&a[i]);
	for(int i=1;i<=m;i++) scanf("%d",&b[i]);
	l=a[1]; r=a[n]; L=l; R=r;
	for(int i=l;i<=r;i++) fa[i]=i;
	for(int i=1;i<=m;i++)
	{
		if(l>0) l-=b[i],r-=b[i];
		else l+=b[i],r+=b[i];
		if(l<=0&&r>=0) 
		{
			int x=R-r,fx=find(x);
			ans[fx]=i;
			if(-l<r)
			{
				for(int i=1;i<=-l;i++) fa[x-i]=x+i,tag[x-i]^=1;
				L=x+1; l=1;
			}
			else
			{
				for(int i=1;i<=r;i++) fa[x+i]=x-i,tag[x+i]^=1;
				R=x-1; r=-1;				
			}
		}
	}
	for(int i=1;i<=n;i++)
	{
		int x=find(a[i]);
		if(ans[x]) printf("Yes %d\n",ans[x]);
		else printf("No %d\n",(r-R+x)*(tag[a[i]]?-1:1));
	}
	return 0;
}

T4 Triangles

这么多天第一次改到 T4!!!

还是 dp,每个顶点记录它向左上,右上,左 延伸的最大距离,首先考虑左和左上的限制如果想延伸一定在这个范围内。

然后对于满足这个限制的点看它向右上延伸距离是否满足。(说不太清,看图更不清

只有三个方向都满足才能转移。定义 \(v1\) 为向右上延伸距离,\(v2\) 为左上延伸最大距离。\(pre\) 为左侧延伸最远的点。

我们考虑每一个点的贡献,也就是每个点会有一个管辖范围。我们分别记它对后面的贡献,和前面哪些值会对它造成贡献。

我们开树状数组,查询 \(x \in [pre,i-1]\) 并且 \(x+v2_x \ge i\)。也就是每一个 \(x+v2_x \ge i\) 都会有贡献。

(下图:黑色是 \(k\) 能为哪些点有贡献,黄色是区间加,蓝色是区间减,红色是统计答案所在区间)

记得反转求一次反向的三角形。

code
#include<bits/stdc++.h>
using namespace std;
const int N = 12005;
#define LL long long
int n,m;
string s[N];
int v1[N>>1][N],v2[N>>1][N],b[N];
LL ans;
struct BIT
{
	int n,c[N];
	void mdf(int x,int v)
	{
		for(;x;x-=(x&-x))  b[x]+=v;
	}
	int que(int x)
	{
		int res=0;
		for(;x<=n;x+=(x&-x)) res+=b[x];//注意树状数组维护的是前缀和,而不是普通的后缀。 
		return res;
	}
	void clear()
	{
		for(int i=1;i<=n;i++) b[i]=0;
	}		
} c;
void reverse()
{
	for(int i=1;i<=(n>>1);i++) swap(s[i],s[n-i+1]);
	for(int i=1;i<=n;i++)
		for(int j=1;j<=m;j++) 
		s[i][j]==0x5c?s[i][j]=0x2f:(s[i][j]==0x2f?s[i][j]=0x5c:0);
}
vector<int> p[N>>2];
vector<pair<int,int> >q[N>>2];
void work(int st)
{
	for(int i=1;i<=n;i+=2)
	{
		c.clear();
		for(int j=(i+st)%4,k=1,pre=1;j<=m;j+=4,k++)
		{
			v2[i][j]=s[i-1][j-1]==0x5c?v2[i-2][j-2]+1:0;
			v1[i][j]=s[i-1][j+1]==0x2f?v1[i-2][j+2]+1:0;
			s[i][j-1]!=0x2d?pre=k:pre;
			p[k].clear(); q[k].clear();
			p[k].push_back(k+v1[i][j]);
			if(max(pre,k-v2[i][j])-1>0) q[max(pre,k-v2[i][j])-1].push_back(make_pair(k,-1));
			if(k-1>0) q[k-1].push_back(make_pair(k,1));
		}
		for(int j=(i+st)%4,k=1;j<=m;j+=4,k++)
		{
			for(int h:p[k]) c.mdf(h,1);
			for(pair<int,int> h:q[k]) ans+=h.second*c.que(h.first);
		}
	}	
}
int main()
{
	scanf("%d%d",&n,&m); c.n=n+m; n=(n<<1)-1; m=(m<<1)-1;
	for(int i=0;i<=n;i++) getline(cin,s[i]),s[i]=' '+s[i];
	work(0);
	reverse();
	work(s[1][1]==0x78?0:2);
	printf("%lld\n",ans);
	return 0;
}
posted @ 2024-07-24 20:19  ppllxx_9G  阅读(12)  评论(0编辑  收藏  举报