首页 > 其他分享 >CodeTON Round 4 (Div. 1 + Div. 2, Rated, Prizes!)(持续更新)

CodeTON Round 4 (Div. 1 + Div. 2, Rated, Prizes!)(持续更新)

时间:2023-04-06 10:26:16浏览次数:44  
标签:Rated const int CodeTON CODE Div include RI define

Preface

唉难得熬夜打一把还天天掉分,苦路西

这把状态奇差,CD两个傻逼题都写的很慢,然后做E的时间太少最后又是经典比赛结束才调出来

虽然很想骂傻逼室友打游戏鬼叫的超级响导致我注意力很难集中,不过终究还是自己抗干扰水平不够,不能怨天尤人


A. Beautiful Sequence

傻逼题,显然若一个位置\(i\)满足\(i\ge a_i\)则通过删除\(i\)之前的某些数一定可以满足题意,直接判断即可

#include<cstdio>
#include<iostream>
#define RI register int
#define CI const int&
using namespace std;
const int N=105;
int t,n,a[N];
int main()
{
	//freopen("CODE.in","r",stdin); freopen("CODE.out","w",stdout);
	for (scanf("%d",&t);t;--t)
	{
		RI i; for (scanf("%d",&n),i=1;i<=n;++i) scanf("%d",&a[i]);
		bool flag=0; for (i=1;i<=n;++i) if (i>=a[i]) flag=1;
		puts(flag?"YES":"NO");
	}
}

B. Candies

首先一个很naive的结论,我们只能生成奇数,因此先特判掉偶数的情况

然后我们手玩一下画一下前几次操作的图出来,就一眼能找到规律,所有数之间形成了一棵二叉树的结果,并且所有数从大到小在树上依次分布

那么这个题就很好处理了,我们把每个奇数的排名(就是第几小的奇数)搞出来做一个类似二进制分解的东西即可

#include<cstdio>
#include<iostream>
#define RI register int
#define CI const int&
using namespace std;
int t,n;
int main()
{
	//freopen("CODE.in","r",stdin); freopen("CODE.out","w",stdout);
	for (scanf("%d",&t);t;--t)
	{
		if (scanf("%d",&n),!(n&1)) { puts("-1"); continue; }
		if (n==1) { puts("0"); continue; }
		RI i,j; for (n=(n-1)/2,i=30;~i;--i) if (n&(1<<i)) break;
		for (printf("%d\n",i+1);~i;--i) printf("%c%c",n&(1<<i)?'2':'1'," \n"[i==0]);
	}
	return 0;
}

C. Make It Permutation

一个朴素的想法,先把所有重复的元素删成一个,然后我们可以直接枚举最后排列的长度,就很容易写出一个式子

其中要插入的次数就是排列的长度\(x\)减去小于等于\(x\)的元素个数,要删除的次数就是大于\(x\)的元素个数

但是有一个问题是最终排列的长度可能是\(10^9\)级别的,不能直接枚举

不过我们对上面的式子稍加观察,我们会发现这个式子要取得最小值只能在“小于等于\(x\)的元素个数”这个值发生改变的时候取

换而言之我们只要枚举所有出现过的数作为最后排列的长度即可,枚举的复杂度就降到\(O(n)\)了

#include<cstdio>
#include<iostream>
#include<algorithm>
#define int long long
#define RI register int
#define CI const int&
using namespace std;
const int N=100005;
int t,n,c,d,a[N]; long long ans;
signed main()
{
	//freopen("CODE.in","r",stdin); freopen("CODE.out","w",stdout);
	for (scanf("%lld",&t);t;--t)
	{
		RI i; for (scanf("%lld%lld%lld",&n,&c,&d),i=1;i<=n;++i) scanf("%lld",&a[i]);
		sort(a+1,a+n+1); int m=unique(a+1,a+n+1)-a-1;
		for (ans=1e18,i=1;i<=m;++i)
		ans=min(ans,1LL*(a[i]-i)*d+1LL*(m-i)*c);
		printf("%lld\n",min(1LL*n*c+d,ans+1LL*(n-m)*c));
	}
	return 0;
}

D. Climbing the Tree

感觉没啥好说的,就一大力模拟题

首先对于每一个\(1\)操作,我们都可以计算出它对应的一个\(h\)的区间,然后我们把这个区间和之前所有\(1\)操作的区间取交集就是现在\(h\)可能的取值区间\([L,R]\)了

具体的计算也很简单,考虑下界就是让蜗牛第\(n-1\)天恰好爬不到,上界就是第\(n\)天刚好爬到,但要注意一些边界情况

然后考虑处理询问,很明显蜗牛要爬的天数关于树高是单调不减的,因此如果对于\(h=L\)和\(h=R\)的树高都需要爬相同的天数那\([L,R]\)之间的任意一天都需要爬相同的天数

然后就模拟一下就完了,只能说这场的CD是真的简单(1300、1700),但我感觉对比周三的那场(1900,2100)我好像写的更久

#include<cstdio>
#include<iostream>
#define int long long
#define RI register int
#define CI const int&
using namespace std;
const int N=100005,INF=1e18;
int t,q,tp,a,b,n;
inline int calc(CI h)
{
	if (a>=h) return 1; return (h-a+(a-b-1))/(a-b)+1;
}
signed main()
{
	//freopen("CODE.in","r",stdin); freopen("CODE.out","w",stdout);
	for (scanf("%lld",&t);t;--t)
	{
		RI i; int L=1,R=INF;
		for (scanf("%lld",&q),i=1;i<=q;++i)
		{
			if (scanf("%lld%lld%lld",&tp,&a,&b),tp==2)
			{
				if (R==INF) { printf("-1 "); continue; }
				int c1=calc(L),c2=calc(R);
				if (c1!=c2) printf("-1 "); else printf("%lld ",c1);
			} else
			{
				scanf("%lld",&n); int l=n>1?(a-b)*(n-2)+a+1:1,r=(a-b)*(n-1)+a;
				if (max(l,L)>min(r,R)) printf("0 "); else L=max(l,L),R=min(r,R),printf("1 ");
			}
		}
		putchar('\n');
	}
	return 0;
}

E. Monsters

唉只能说我想题和写题的速度都还是有点慢,每次比赛都有一道题要血战到最后一刻

虽然以前有几次也能堪堪压时过,但分数也是寥寥无几了,这次还比较可惜没调出来(其实就是个傻逼下标写反了)

好了说回题目,这题我本来的思路是按照每条边两端的点权值的差的绝对值从小到大排序

然后每次处理一条边的时候就尝试合并,通过并查集来维护每个连通块的大小,看起来就挺正确

后面写完调试样例的时候感觉不对劲,有些比如联通了权值为\((3,3)\)的边显然不应该先处理,因为它们还有等一些比如\((1,2)\)这样的边处理之后再联通的机会

因此后面改成了按照每条边两端的点的权值的最大值从小到大来排序就很正确了,代码也是十分好写

#include<cstdio>
#include<iostream>
#include<algorithm>
#define RI register int
#define CI const int&
using namespace std;
const int N=200005;
int t,n,m,a[N],x,y,fa[N],sz[N],vis[N];
struct edge
{
	int v,x,y;
	friend inline bool operator < (const edge& A,const edge& B)
	{
		return A.v<B.v;
	}
}e[N];
inline int getfa(CI x)
{
	return fa[x]!=x?fa[x]=getfa(fa[x]):x;
}
int main()
{
	//freopen("CODE.in","r",stdin); freopen("CODE.out","w",stdout);
	for (scanf("%d",&t);t;--t)
	{
		RI i; for (scanf("%d%d",&n,&m),i=1;i<=n;++i)
		scanf("%d",&a[i]),fa[i]=i,sz[i]=1,vis[i]=a[i]==0;
		for (i=1;i<=m;++i) scanf("%d%d",&e[i].x,&e[i].y),e[i].v=max(a[e[i].y],a[e[i].x]);
		for (sort(e+1,e+m+1),i=1;i<=m;++i)
		{
			int fx=getfa(e[i].x),fy=getfa(e[i].y);
			if (fx==fy) continue; 
			vis[fx]=((vis[fx]&&sz[fx]>=e[i].v)||(vis[fy]&&sz[fy]>=e[i].v));
			fa[fy]=fx; sz[fx]+=sz[fy];
		}
		puts(sz[getfa(1)]==n&&vis[getfa(1)]?"YES":"NO");
	}
}

后来ztc跟我讲了一个BFS的做法啊,但是他的做法一些维护细节我也不是很理解,不过后来顺着想想发现了我的另一种实现方法

就是我们从所有\(a_i=0\)的点为源点做BFS,然后每次向外走之后可以把当前走到的所有点合并看作一个大的连通块来看待

然后考虑对于每个连通块我们每次要找到这些点向外的连边可达的节点中,\(a_i\)最小的那个来扩展,如果找得到并且这个值小于等于当前连通块就扩展,否则这个连通块就没用了

我们考虑对每一个连通块用堆维护它出点的\(a_i\),然后考虑在合并的时候用启发式合并合并两个堆,最后由于查询次数是\(O(n)\)的,所以均摊复杂度是\(O(n\log n)\)的

代码就没写了,口胡一下

标签:Rated,const,int,CodeTON,CODE,Div,include,RI,define
From: https://www.cnblogs.com/cjjsb/p/17291786.html

相关文章

  • 8.1 js addEventListener js给div添加事件
    <!DOCTYPEhtml><html><head><metacharset="utf-8"><title>testaddEventListener</title></head><body><buttonid="myBtn">clickme</button><pid="demo">&......
  • Codeforces Round 642 (Div3)
    K-periodicGarland给定一个长度位\(n\)的\(01\)串,每次操作可以将\(1\)变为\(0\)或者将\(0\)变为\(1\),现在你需要通过操作使得所有\(1\)之间的距离为\(k\),求最少的操作次数,注意全为\(0\)也算\(1<=n<=1e6,1<=k<=n\)\(dp\)/贪心:最大子段和思想方法一:\(dp\)\(O(n)\)状......
  • Codeforces Round 862 (Div. 2)
    CodeforcesRound862(Div.2)链接CodeforcesRound862(Div.2)A题#include<iostream>#include<algorithm>#include<cstdio>#include<cstring>#include<vector>#include<cstring>#include<unordered_set>#includ......
  • Codeforces Round 863 (Div. 3)
    CodeforcesRound863(Div.3)链接CodeforcesRound863(Div.3)A题遍历这个字符串,如果要插入的数第一次小于当前的数,就将数插入到这里,如果到最后都没有插入数,插入到最后#include<iostream>#include<algorithm>#include<cstdio>#include<cstring>#include<vec......
  • Codeforces Round 863 (Div. 3) E题
    题目地址题意:定义数组a包含所有不含数字4的正整数,给出一个n,要求求出数组a中第n个数Solution数位dp+二分,求出[1,mid]中不含数字4的正整数个数,不过因为有可能mid包含4,但是由于贡献是一样的,可以直接把4都变成3,最后处理一下即可intdp[20];inta[20];voidinit(){ dp[0]=1; f......
  • Codeforces Round 863 (Div. 3)
    A.InsertDigit放在第一个比他小的数前面#include<bits/stdc++.h>usingnamespacestd;voidsolve(){intn,d;cin>>n>>d;strings;cin>>s;for(chari:s){if(d>i-'0')cout<<d,d......
  • Codeforces Round 640 (Div. 4) ABCDEFG
    https://codeforces.com/contest/1352不知道怎么的复制过来的代码容易歪,观看效果可能不大好。这场古早div4,大题极其友好,除了E卡空间卡到我爆炸,别的都体验感极好。A.SumofRoundNumbers#include<bits/stdc++.h>usingnamespacestd;typedeflonglongLL;typedefpai......
  • Codeforces Round 863 (Div. 3) A-C 赛后思路复盘
    A(思维)思路:观察样例可知数越大放在前面越优。遍历字符串,判断当前位置的数字和要插入的数字的关系,如果要插入的数大于当前数,那么就插入到当前数的前面。string里有一个insert函数,可以把指定字符串插入到指定下标之前。在原串下标为pos的字符前插入字符串strbasic_string&insert......
  • cf-div.3-863d
    题目链接:https://codeforces.com/contest/1811/problem/D思维题,昨天被E题搞太久了,这题认真想的话应该可以出的。思路:不断循环,判断x和y是否在合法区间内。代码:#include<bits/stdc++.h>usingnamespacestd;constintN=2e5+10;longlongfib[70];voidsolve(){int......
  • Codeforces Round 861 (Div. 2)
    Preface这场感觉都是一个礼拜前补题打的了,但由于上周末事情比较多都没来得及写题解因此可能题意都记得不是很清楚了,就简略地谈一谈吧A.LuckyNumbers不难想到直接暴力从左端点枚举到右端点并对每个数进行暴力判断一个很naive的结论就是当答案为\(9\)时直接输出即可,然后我们......