Educational Codeforces Round 161 (Rated for Div. 2)补题

文章介绍了四个Codeforces问题的解题思路,涉及模板字符串匹配、二进制分析、动态规划在形成三角形和子序列问题中的应用,以及处理城市间的最短路径问题。
摘要由CSDN通过智能技术生成

Tricky Template(Problem - A - Codeforces

题目大意:现有三个模板字串a,b,c,都由小写字母组成,问能否找到一个字串s,使s与a,b匹配,与c不匹配,匹配的条件如下:

如果s的某位为小写字母,那么需要与匹配串的对应位完全相同

如果某位为大写字母,那么需要与匹配串的对应位不同

思路:这题的考点就在于什么时候出现题目要求的情况,只要有一位满足,哪怕剩下的不满足也无所谓,那么就要看a[i],b[i],c[i]的关系,如果a[i]=b[i]=c[i],那么这一位能与a,b匹配就一定能与c匹配,如果a[i]=b[i]!=c[i],那么填上a[i]即可,如果如果a[i]!=b[i],这一位可以填一个大写字母,这个大写字母与c[i]相同但不与a[i],b[i]相同即可,也就是说,只要有一个位置a[i]!=c[i]&&b[i]!=c[i]就成立。

#include<bits/stdc++.h>
using namespace std;
int main()
{
	int t;
	scanf("%d",&t);
	while(t--)
	{
		int n;
		string a,b,c;
		cin>>n>>a>>b>>c;
		int flag=0;
		for(int i=0;i<n;i++)
		{
			if(a[i]!=c[i]&&b[i]!=c[i]) 
			{
				flag=1;
				break;
			}
		}
		if(flag) printf("YES\n");
		else printf("NO\n");
	}
}

Forming Triangles(Problem - B - Codeforces

题目大意:现给出若干条边,每条边的长度为2^a[i],要找出三边组成一个三角形(非退化三角形就是一个有面积的三角形),问最多有多少种组合(数相同,但所处位置不同视为不同的选择)

思路:这里提到了2^a[i],那么就要从二进制的角度来看看有什么规律:

2^i=2^(i-1)+2^(i-1) 

 然后又由三角形两边之和大于第三边,所以我们必须选两个相同的,然后再选一个小于等于它们的。那么情况实际就出来了,先统计出每种长度边的个数,用map统计,刚好还能顺便排个序。然后从小到大访问,对于每种长度,如果大于等于2才有讨论的价值,选出两条当前长度的边,假设当前长度的有k条边,那么这两条边的选择的情况就有(k-1)*k/2,然后第三条边的选择为前c条边的总个数+(k-2),那么遍历map就能统计出来。

#include<bits/stdc++.h>
using namespace std;
#define int long long
signed main()
{
	int t;
	scanf("%lld",&t);
	while(t--)
	{
		int n;
		scanf("%lld",&n);
		map<int,int>mp;
		for(int i=1;i<=n;i++) 
		{
			int x;
			scanf("%lld",&x);
			mp[x]++;
		}
		int c=0;
		int ans=0;
		for(auto it:mp)
		{
			int d=it.second;
			if(d>=2)
			{
				ans += (d-1)*d/2*c;//前面的
				ans += (d-2)*(d-1)*d/6;//本身
			}
			c+=d;
		}
		printf("%lld\n",ans);
	}
}

Closest Cities(Problem - C - Codeforces

题目大意:现有一排城市,它们的坐标单增,然后每个城市有一个最近城市(距离绝对值是这个城市到所有城市中最小的),对于城市x,如果要到城市y,那么有两种情况:

y是x的最近城市:花费1;

y不是x的最近城市:花费abs(ax-ay)

现有q个查询,a,b,求a到b的最小花费。

思路:我们来看从a到b,如果直接走,由于坐标单增,实际上相当于一个点一个点走,两点之间花费为坐标差,或者1,因为单增,所以不会重复计算某一段。然后有q个询问,那么我们只要预处理出来从前到后和从后到前的前缀和不就好了。

#include<bits/stdc++.h>
using namespace std;
int a[200010],ne[200010],s1[200010],s2[200010];
signed main()
{
	int t;
	scanf("%d",&t);
	while(t--)
	{
		int n;
		scanf("%d",&n);
		for(int i=1;i<=n;i++) scanf("%d",&a[i]);
		ne[1]=2,ne[n]=n-1;
		for(int i=2;i<=n-1;i++)
		{
			int l=a[i]-a[i-1],r=a[i+1]-a[i];
			if(l<r) ne[i]=i-1;
			else ne[i]=i+1;
		}
		s1[1]=0;
		int c=1;
		for(int i=2;i<=n;i++)
		{
			if(ne[c]==i) s1[i] = s1[c]+1;
			else s1[i] = s1[c]+a[i]-a[c];
			c=i;
		}
		s2[n]=0;
		c=n;
		for(int i=n-1;i>=1;i--)
		{
			if(ne[c]==i) s2[i]=s2[c]+1;
			else s2[i]=s2[c]+a[c]-a[i];
			c=i;
		}
		int q;
		scanf("%d",&q);
		while(q--)
		{
			int l,r,ans=0;
			scanf("%d%d",&l,&r);
			if(l<r)
				ans = s1[r]-s1[l];
			else
				ans=s2[r]-s2[l];
			printf("%d\n",ans);
		}
	}
}

 ps:这里我最初还想的是给每个点和它可以到的点打上标记,然后判断标记是否相同(来自Salyg1n and the MEX Game(Problem - C - Codeforces),包括画的有向图的想法也来自这里),然后后来又想到遍历每个点,以每个点作为中转(来自最长上升子序列状态转移的思路),再后来又想到找离它们最近的标记相同的点(来自2D Traveling(Problem - B - Codeforces)),再往后发现找起来太麻烦,然后发现性质,打算遍历,然后超时了,想到前缀和优化,然后ac。所以,思路未必是一次就想到的,就是这样不断猜测不断尝试不断推翻最后实现的过程。而且这每一步其实都是联想到之前题目的解法类比出来的,所以呀,哪有什么捷径,都是不断学习新的知识点,不断补题学习新思路,不断内化,然后转为输出的过程。

Berserk Monsters(Problem - D - Codeforces

题目大意:现在有n个怪兽排成一排,每个怪兽有一个攻击值a和一个防御值d,对它们发动魔法攻击,它们可以同时对相邻位置的怪兽产生a的伤害,如果一个怪兽受到的伤害值大于d,那么这个怪兽就死亡,游戏进行n轮,要求每轮死亡怪兽的数目。

思路:乍一看很简单,就是一个双向链表的题,可以定义l[],r[]两个数组来表示指针,单独定义一个数组储存它们受到的伤害,最后链表遍历结束时,遍历怪兽,将受到伤害的怪兽移出链表。看起来很对,也确实可以实现,但是这里的时间复杂度就是O(n^2),因为回合是n轮,每次遍历怪兽判断是否死亡的时间复杂度也是O(n),那么合起来就是O(n^2),有没有什么办法可以进行优化呢?显然是有的,我们可以发现怪兽左边相邻的怪兽的下标一定比它小,右边相邻的怪兽的下标一定比它大。然后我们与删除位置相邻的怪兽的左右会变,别的怪兽的左右不会变,所以它原来如果没被删除,那么将可以删除的部分删除后,它们仍然不会被删除,那么就没有遍历的必要。所以我们将删除位置左右没被删除的元素记录下来即可。再次遍历时,就只访问这些元素。然后再来看,我们如何去删除元素呢,显然用:

r[l[i]]=r[i]

l[r[i]]=l[i]

那么我们如果有三个元素a,b,c,相邻的两个元素b,c都被删除,岂不是a被被放进新数组两次,这也产生了冗余,那就需要用到vector容器的两个函数unique(),resize(),unique()将所有元素调整一下,对于重复元素,取一个放到前面来,剩下的都放在后面,结束时,返回不重复部分尾元素的迭代器,resize()重新分配空间大小,所以就是g.resize(unique(g.begin(),g.end())-g.begin())即可实现,那么这道题的优化就完成了,我们只记录删除位置左右未被删除的元素,所以最坏情况是隔一个删一个那么每次砍半,但是并不会真的出现这种情况,所以时间复杂度是小于O(nlogn)的,可以AC。

#include<bits/stdc++.h>
using namespace std;
int a[300010],d[300010],l[300010],r[300010],st[300010];
int main()
{
	int t;
	scanf("%d",&t);
	while(t--)
	{
		int n;
		scanf("%d",&n);
		vector<int>g,v;
		for(int i=1;i<=n;i++) 
		{
			scanf("%d",&a[i]);
			g.push_back(i);
			l[i]=i-1;
			r[i]=i+1;
			st[i]=1;
		}
		a[n+1]=0;
		r[n]=0;
		for(int i=1;i<=n;i++) scanf("%d",&d[i]);
		for(int k=0;k<n;k++)
		{
			sort(g.begin(),g.end());//O(mlogm),m<=n
			g.resize(unique(g.begin(),g.end())-g.begin());O(m),m<=n
			for(auto it:g)O(m),m<=n
			{
				if(d[it]<a[l[it]]+a[r[it]])
				{
					v.push_back(it);
					st[it]=0;
				}
			}
			g.clear();
			for(auto i:v)//O(m),m<=n
			{
				/*这里实际上只考虑原来位置删除后,新产生的相邻关系会不会造成删除,原来不能删的,如果相邻元素没变,
				一样不能删,没有考虑的必要,原来能删的已经删除,自然不会被放入,所以g一下被缩小到很小,
				最坏隔一位删一个,那么就变成一半,剩余的都比一半小,所以最坏是logn,而且肯定不到logn*/
				l[r[i]]=l[i],r[l[i]]=r[i];
				if(st[l[i]]) g.push_back(l[i]);
				if(st[r[i]]) g.push_back(r[i]);
 			}
 			cout<<v.size()<<" ";
 			v.clear();
		}
		cout<<endl;
	}
}

Increasing Subsequences(Problem - E - Codeforces

题目大意:现在要创建一个数组,使数组中的严格上升子序列的个数恰好为x个。输出数组长度和具体的数,如果不成立则输出-1.另外数组长度不能超过200。

思路:这个题按照题目意思来看,这个数组可以是任意的,所以正着想不大好想,似乎没什么规律,那么我们倒着来看,如何统计一个序列中单增序列的个数,这里显然需要用到动态规划:

dp[i]表示以i作为结尾的单增序列的个数,那么就如下更新:

dp[i]=1(前面一个序列都不加)

dp[i]+=dp[x](1<=x<v,如果a[i]>a[x])

我们就此找出每个位置可以产生的上限,进而算出总的最大值,因为要考虑是否可以产生这样的数组

max:

dp[1]=1;

dp[2]=1+dp[1]=2;

dp[3]=1+dp[1]+dp[2]=4;

dp[4]=1+dp[1]+dp[2]+dp[3]=8;

... 

 是不是有点特点,某个位置前面如果有多少个小于它的数,那么就可以产生2多少次方个贡献。

所以我们考虑将n转化成二进制的角度来看,我最开始的思路是通过将高位转化到低位,然后按照

v,v+1,...,v+k,v,v+1,v+2,...

的顺序来放,每一组内部包含一个所有位。

但是很显然长度会超,那么我们在找找别的规律来看:

以37为例:

100101

空也为一种情况,所以凑出36即可

100100

如果将最高位拆下来:011212

再举一个偶数来看:

36

则需凑出35:

100011

如果将最高位拆下来:011123;

我们会发现拆的时候除了最后一位和原本是1的地方外,其他都是1,为1则表示前面有多少个小于它的数,那么我们可以设定一个单增的序列:

1 2 3 4 5 5

那么得到的单增序列长度为:1 2 4 8 16 16

这样可以看出,1和2都能对应凑出来,如果为3的话,也是最后一位为3,前面有0个小于它的数,那么直接开头填一个最大值即可。

那么思路就有了,我们直接拆太麻烦,就看原数的每位数的前一位(从右往左访问),如果为1,那么在添完本位数后,后面多填一个最大的数,这样就可以保证下一位有两个。这样看实际0位,如果原数是偶数,那么0位就是0,最后应该有两个数前面有0个元素小于它,一个是我们填进去的第一个元素,一个代表空集合。如果原数是奇数,那么这一位实际应该有3个数前面有0个元素小于它,除了刚提到的两个,我们再多加一个最大数即可。

#include<bits/stdc++.h>
using namespace std;
#define int long long
signed main()
{
	int t;
	scanf("%lld",&t);
	while(t--)
	{
		int n;
		scanf("%lld",&n);
		int k=__lg(n);//二进制下最高位1在哪里
		vector<int>g;
		if(n%2) g.push_back(k);//出了空还多一个1
		for(int i=0;i<k;i++)
		{
			g.push_back(i);
			if(n>>i+1&1 && i+1!=k) g.push_back(k);
		}
		cout<<g.size()<<endl;
		for(auto i:g) cout<<i<<" ";
		cout<<endl; 
	}
}

 ps:哈哈哈,这次终于轮到我卡着最后一分钟ac了。所以哪有什么应不应该,没思路就去想思路,有思路了就去实现思路,那么ac是必然的事情,只要还没结束,那么就不要认输放弃,说不定下一秒就写出来了。

  • 26
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
"educational codeforces round 103 (rated for div. 2)"是一个Codeforces平台上的教育性比赛,专为2级选手设计评级。以下是有关该比赛的回答。 "educational codeforces round 103 (rated for div. 2)"是一场Codeforces平台上的教育性比赛。Codeforces是一个为程序员提供竞赛和评级的在线平台。这场比赛是专为2级选手设计的,这意味着它适合那些在算法和数据结构方面已经积累了一定经验的选手参与。 与其他Codeforces比赛一样,这场比赛将由多个问题组成,选手需要根据给定的问题描述和测试用例,编写程序来解决这些问题。比赛的时限通常有两到三个小时,选手需要在规定的时间内提交他们的解答。他们的程序将在Codeforces的在线评测系统上运行,并根据程序的正确性和效率进行评分。 该比赛被称为"educational",意味着比赛的目的是教育性的,而不是针对专业的竞争性。这种教育性比赛为选手提供了一个学习和提高他们编程技能的机会。即使选手没有在比赛中获得很高的排名,他们也可以从其他选手的解决方案中学习,并通过参与讨论获得更多的知识。 参加"educational codeforces round 103 (rated for div. 2)"对于2级选手来说是很有意义的。他们可以通过解决难度适中的问题来测试和巩固他们的算法和编程技巧。另外,这种比赛对于提高解决问题能力,锻炼思维和提高团队合作能力也是非常有帮助的。 总的来说,"educational codeforces round 103 (rated for div. 2)"是一场为2级选手设计的教育性比赛,旨在提高他们的编程技能和算法能力。参与这样的比赛可以为选手提供学习和进步的机会,同时也促进了编程社区的交流与合作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值