AFO

CF1173X

CF1173C

由于牌堆只能从最后插牌,所以插牌方法非常显然

首先特判一下牌堆有没有一个合法的后缀,如果有的话再判断一下手中的牌和合法后缀之前的牌的排列顺序能不能有效的继续续下去

然后排除了以上情况就可以直接找\(max(i-b_i)\)就是最早在哪个时刻开始往里按顺序加牌构造递增序列

#include<iostream>
#include<cstdio>

using namespace std;

int w,n,m,k,a[1000001],b[1000001],c[300001];

int main()
{
	scanf("%d",&n);
	for(int i=1;i<=n;i++) scanf("%d",&b[i]);
	for(int i=1;i<=n;i++) scanf("%d",&a[i]), c[a[i]]=i;
	int B=0;
	k=n;
	while(k && a[k-1]==a[k]-1) k--;
	if(a[k]==1 || a[k]==0) B=1;
	if(a[n]==0) B=1;
	if(B)
	{
		for(int i=1;i<k;i++)
			if(a[i] && i-a[i]+a[n]+1>0) B=0;
		if(B) 
		{
			printf("%d",n-a[n]);
			return 0;
		}
		
	}
	B=0;
	for(int i=1;i<=n;i++) if(a[i])B=max(B,i-a[i]+1);
	printf("%d",B+n);
}

CF1173D

边不交叉的条件是每棵子树都要在一段连续的位置

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define LL long long 
using namespace std;
const int M = 1000001;
const int P = 998244353;

int x,y,n,m,k, ver[M], nex[M], edge[M],cnt,A[M],head[M];

void add(int x,int y)
{
	ver[++cnt]=y, nex[cnt]=head[x], head[x]=cnt;
	ver[++cnt]=x, nex[cnt]=head[y], head[y]=cnt;
}

int dfs(int x,int fa)
{
	int s=1, d=1;
	for(int i=head[x];i;i=nex[i])
	{
		if(ver[i]==fa) continue;
		d=(LL)d*dfs(ver[i],x)%P;
		s++;
	}
	if(x!=1)return (LL)A[s]*d%P;
	return (LL)A[s-1]*d%P;

}

int main()
{
	scanf("%d",&n);
	for(int i=1;i<n;i++)
	{
		scanf("%d%d",&x,&y);
		add(x,y);
	}
	A[0]=1;
	for(int i=1;i<=n;i++) A[i]=(LL)A[i-1]*i%P;
	printf("%d",(LL)n*dfs(1,0)%P);
}

posted @ 2019-09-16 11:37  ZUTTER☮  阅读(123)  评论(0编辑  收藏  举报