• CF# div.2赛后总结


    前言

    比赛链接

    ZWQking AK啦!!!!!!!Orz

    A

    题意:有(n)个小孩,有(4n)个位置,要求你安排小孩坐位置,使得被坐的位置的编号(a,b)满足:(gcd(a,b)≠1,a,b)

    做法:构造法,让他们坐(2n+2,2n+4,2n+6,...,4n)的位置即可。

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    #include<cmath>
    using  namespace  std;
    int  main()
    {
    	int  T;scanf("%d",&T);
    	while(T--)
    	{
    		int  n;scanf("%d",&n);
    		int  limit=4*n,ed=2*n;
    		for(int  i=limit;i>ed;i-=2)printf("%d ",i);
    		printf("
    ");
    	}
    	return  0;
    }
    

    B

    题意:给你一个字符串,(1)的位置有地雷,(0)没有,你可以花(a)代价引爆连续一段的雷,或者花(b)代价埋一颗雷。

    做法:

    1. DP做法,(dp[i][0/1])分别表示这个位置没有地雷和有地雷的转移。
    2. 赛后想了想,可以贪心,计算中间的每一段连续的(0)填满的代价,如果小于(a)则填满,否则不填满,直接两端引爆。

    我才用的是(DP)做法。

    两种做法时间复杂度都是:(O(n))

    #include<cstdio>
    #include<cstring>
    #define  N  110000
    using  namespace  std;
    inline  int  mymin(int  x,int  y){return  x<y?x:y;}
    inline  int  mymax(int  x,int  y){return  x>y?x:y;}
    int  dp[N][2];
    char  st[N];
    int  n,a,b;
    int  main()
    {
    	int  T;scanf("%d",&T);
    	while(T--)
    	{
    		scanf("%d%d",&a,&b);
    		scanf("%s",st+2);
    		n=strlen(st+2)+2;
    		st[1]=st[n]='0';
    		for(int  i=1;i<=n;i++)dp[i][0]=dp[i][1]=999999999;
    		dp[1][0]=0;
    		for(int  i=2;i<=n;i++)
    		{
    			if(st[i]=='0')
    			{
    				dp[i][0]=mymin(dp[i-1][0],dp[i-1][1]+a);
    				dp[i][1]=mymin(dp[i-1][0],dp[i-1][1])+b;
    			}
    			else  dp[i][1]=mymin(dp[i-1][0],dp[i-1][1]);
    		}
    		printf("%d
    ",dp[n][0]);
    	}
    	return  0;
    }
    

    C

    题意: 给你(a)数组和(b)数组,对于每个(i),要么让小明多花(b_{i})的时间,要么直接多出一个人花(a_{i})的时间。

    然后问你如何安排才可以让花费时间最大的人最小。

    做法:二分答案,还算简单。

    #include<cstdio>
    #include<cstring>
    #define  N  210000
    using  namespace  std;
    inline  int  mymax(int  x,int  y){return  x>y?x:y;}
    int  a[N],b[N],n;
    inline  bool  check(int  k)
    {
    	int  shit=0;
    	for(int  i=1;i<=n;i++)
    	{
    		if(a[i]>k)
    		{
    			shit+=b[i];
    			if(shit>k)return  0;
    		}
    	}
    	return  1;
    }
    int  main()
    {
    	int  T;scanf("%d",&T);
    	while(T--)
    	{
    		scanf("%d",&n);
    		int  l=1,r=0;
    		for(int  i=1;i<=n;i++)
    		{
    			scanf("%d",&a[i]);
    			r=mymax(r,a[i]);
    		}
    		for(int  i=1;i<=n;i++)scanf("%d",&b[i]);
    		int  mid,ans=r;
    		while(l<=r)
    		{
    			mid=(l+r)>>1;
    			if(check(mid)==1)r=mid-1,ans=mid;
    			else  l=mid+1;
    		}
    		printf("%d
    ",ans);
    	}
    	return  0;
    }
    

    D

    题意: 给你一个(a)数组,你有两种操作:

    1. 选择一个(i),让(1)~(i)的位置全部减一。
    2. 选择一个(i),让(i)~(n)的位置全部减一。
      问是否可以把(a)数组全部变成(0)

    做法:他们都说很简单,就我想了挺久的。

    我们不妨处理(1)操作,设(b)数组,(b_{i})表示第(i)个位置经过(1)操作减去了(b_{i})

    (b_{i}≥b_{i+1})

    这样,只需要构造一个合法的(b)数组使得(a)数组减完(b)数组后从右往左到最后一个非(0)数字非严格单调递减。

    然后就开始考虑差分约束,(b_{i}≥b_{i+1})(i+1)(i)连一条(0)的边,因为要(a_{i}-b_{i}≤a_{i+1}-b_{i+1}),所以(a_{i}-a_{i+1}+b_{i+1}≤b_{i})(i+1)(i)连接一条边权(a_{i}-a_{i+1})的边,从(n)点开始跑最长路,然后最后检查(b_{i}≤a_{i})即可,但是后面发现了一个事情,边只会从(i+1)连向(i),直接一遍扫过去就行了啊(╯‵□′)╯︵┻━┻。

    当然,你可以直接默认(b_{n})(0),因为如果(b_{n}>0),完全可以把(1)~(n)(1)操作拆成一个(1)操作一个(2)操作来搞,所以可以直接默认(b_{n}=0)

    时间复杂度:(O(n))

    #include<cstdio>
    #include<cstring>
    #include<queue>
    #define  N  31000
    using  namespace  std;
    int  dp[N],n,a[N];
    inline  int  mymax(int  x,int  y){return  x>y?x:y;}
    int  main()
    {
    	int  T;scanf("%d",&T);
    	while(T--)
    	{
    		scanf("%d",&n);
    		for(int  i=1;i<=n;i++)scanf("%d",&a[i]);
    		bool  bk=0;
    		dp[n]=0;
    		for(int  i=n-1;i>=1;i--)
    		{
    			if(a[i]>a[i+1])dp[i]=dp[i+1]+a[i]-a[i+1];
    			else  dp[i]=dp[i+1];
    			if(dp[i]>a[i])
    			{
    				bk=1;
    				break;
    			}
    		}
    		if(!bk)printf("YES
    ");
    		else  printf("NO
    ");
    	}
    	return  0;
    }
    

    E

    题意:默认现在是字典序最小的全排列(即:(1,2,3,4,...,n)),长度为(n),然后有两个操作:

    1. 统计([l,r])的区间和。
    2. 假设现在的全排列字典序排名为(x),给你一个(y),让你把全排列变成字典序排名为(x+y)的全排列。

    做法: 和康托展开非常有关系,因为全排列排名总和为(1e12),发现(16!)已经大于这个数字,暴力维护后面(16)个数字,然后暴力统计即可。

    时间复杂度:(O(q16^2))

    #include<cstdio>
    #include<cstring>
    #include<cstdlib>
    #include<algorithm>
    #define  N  210000
    #define  NN  410000
    using  namespace  std;
    typedef  long  long  LL;
    int  a[N],b[20],n,q;
    LL  fc[20];
    bool  v[N];
    void  kangtuo(int  l,LL  k)
    {
    	int  top=0;
    	for(int  i=l;i<=n;i++)b[++top]=i,v[top]=0;
    	sort(b+1,b+top+1);
    	for(int  i=top;i>=2;i--)
    	{
    		LL  shit=k/fc[i-1];k%=fc[i-1];
    		for(int  j=1;j<=top;j++)
    		{
    			if(!shit  &&  !v[j])
    			{
    				a[top-i+l]=b[j];
    				v[j]=1;
    				break;
    			}
    			if(!v[j])shit--;
    		}
    	}
    	for(int  i=1;i<=top;i++)if(!v[i])a[n]=b[i];
    }
    LL  nowcnt=0;
    inline  LL  getsum(int  l,int  r){return  (LL)(l+r)*(r-l+1)/2;}
    int  main()
    {
    	scanf("%d%d",&n,&q);
    	fc[0]=1;for(int  i=1;i<=16;i++)fc[i]=fc[i-1]*(LL)i;
    	for(int  i=1;i<=n;i++)a[i]=i;
    	
    	int  ll=n-16+1;
    	if(ll<=0)ll=1;
    	for(int  i=1;i<=q;i++)
    	{
    		int  type;scanf("%d",&type);
    		if(type==1)
    		{
    			int  l,r;scanf("%d%d",&l,&r);
    			if(r<ll)printf("%lld
    ",getsum(l,r));
    			else
    			{
    				LL  sum=0;
    				int  lll=ll;
    				if(l>=ll)lll=l;
    				else  sum=getsum(l,ll-1);
    				for(int  i=lll;i<=r;i++)sum+=a[i];
    				printf("%lld
    ",sum);
    			}
    		}
    		else
    		{
    			int  x;scanf("%d",&x);
    			nowcnt+=x;
    			kangtuo(ll,nowcnt);
    		}
    	}
    	return  0;
    }
    

    F

    题意:给你一个(a)数组,长度为(n),可以操作(k)次,第(t)次操作,你可以删掉第(i(1≤t≤n-t+1))个数字,然后把(a_{i+1})或者(a_{i-1})(必须满足被贴的数字有意义,即在数组范围内,且必须贴数字)贴到(b)数组最右边((b)数组一开始为空),然后把(a_{i})~(a_{n})全部往左移一位,现在给你(a,b)数组((a,b)数组的数字都是不同的),问你操作序列能有多少个。

    做法: 首先化一下题意:(b)数组在(a)数组中对应的位置被锁上了,也就是不能被删除,然后对于(b_{1}),其在(a)数组对应的位置为(a_{i}),如果(i+1,i-1)的位置都被锁上了,这个(b)数组绝对得不到,否则,假设(i-1)解锁了,相当于牺牲掉(i-1)的位置给(i)位置解锁。

    在上述过程中,不难发现,如果对于一个上锁的位置(i),左边如果没有被锁,或者解锁的时间早于它,那么在解锁(i)时,左边一定是未锁上的位置(因为如果左边牺牲自己去解锁更左边的位置,那么更左边就变成了未锁上的左边),右边同理。

    在这里插入图片描述
    橙色为锁上的位置,红色为未锁上的位置,蓝色代表牺牲。

    这样,就只需要在开始的时候判断每个位置是否可以牺牲左边或者右边即可。

    时间复杂度:(O(n))

    #include<cstdio>
    #include<cstring>
    #define  N  210000
    using  namespace  std;
    typedef  long  long  LL;
    const  LL  mod=998244353;
    int  a[N],b[N],c[N],n,k;
    inline  LL  ksm(LL  x,LL  y)
    {
    	LL  ans=1;
    	for(LL  i=1;i<=y;i++)ans=ans*x%mod;
    	return  ans; 
    }
    int  main()
    {
    	int  T;scanf("%d",&T);
    	while(T--)
    	{
    		scanf("%d%d",&n,&k);
    		for(int  i=1;i<=n;i++){scanf("%d",&a[i]);c[i]=0;}
    		int  cnt=0;LL  ans=0;
    		for(int  i=1;i<=k;i++)
    		{
    			scanf("%d",&b[i]);
    			c[b[i]]=i;
    		}
    		for(int  i=1;i<=n;i++)
    		{
    			if(c[a[i]]>0)
    			{
    				if((c[a[i]]<c[a[i+1]]  &&  c[a[i]]<c[a[i-1]])  ||  (i==1  &&  c[a[i+1]]>c[a[i]])  ||  (i==n  &&  c[a[i-1]]>c[a[i]]))
    				{
    					ans=-1;
    					break;
    				}
    				else  if(i!=1  &&  i!=n  &&  c[a[i-1]]<c[a[i]]  &&  c[a[i+1]]<c[a[i]])cnt++;
    			}
    		}
    		if(ans==-1)printf("0
    ");
    		else  printf("%lld
    ",ksm(2,cnt));
    	}
    	return   0;
    }
    
  • 相关阅读:
    USACO2008 Cow Cars /// oj23323
    USACO2008 Roads Around The Farm /// queue oj23321
    USACO2007 捕牛记 /// queue+桶 oj1503
    哈理工多校算法赛一
    USACO2012 Haybale stacking /// 区间表示法 oj21556
    USACO2012 Broken necklace /// DP oj10103
    USACO2004 cube stacking /// 带权并查集 oj1302
    ACM-ICPC 2018 南京赛区网络预赛 J sum (找一个数拆成两个无平方因子的组合数)
    分层图 (可以选择K条路的权为0,求最短路)
    POJ 3537 Crosses and Crosses(sg博弈)
  • 原文地址:https://www.cnblogs.com/zhangjianjunab/p/13918169.html
Copyright © 2020-2023  润新知