Codeforces Round 922 (Div. 2)补题

Brick Wall(Problem - A - Codeforces

题目大意:规定砖的大小为1*k(k>=2),现在有一面n*m的砖墙,n是墙高,m是墙宽,砖在砖墙中有两种放法,水平放置和竖直放置,墙的稳定性=水平放置的砖的数量-竖直放置的砖的数量。现在要求墙的稳定性的最大值。注意墙不用填满,砖不能相交。

思路:很显然我们尽可能地水平放置,不竖直放置即可。反正又不用填满。

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
int main()
{
	int t;
	scanf("%d",&t);
	while(t--)
	{
		int n,m;
		scanf("%d%d",&n,&m);
		m /= 2;
		n*=m;
		printf("%d\n",n);
	}
}

Minimize Inversions(Problem - B - Codeforces

题目大意:现在有两个排列a,b,我们需要选择两个位置i,j,交换ai,aj和bi,bj,最后要使两个序列中的反转数最小。

思路:刚开始我看到这个题也没有什么头绪,然后就去分析了下样例,前两个样例没什么感觉,看到第三个样例的时候,我突然发现第三个样例的输出中第二个排列有序的。呀,昨晚上太着急了,以为它们是有序的,刚刚一看竟然不是,但是最后的结果确实是使一个排序变成有序的。不过我们还是来证明一下比较放心。

我们很容易通过交换使一个排列变成有序的,这样的话,两个排列中的反转数会是最小的吗?显然在有序的那个排列中反转数为0,那么在另一个序列中呢?哎,这个有点证不明白,但是写题解还是不要水,我去看了下官方的证明,思路大概是这样:

对于a,b我们选出来的i,j构成的逆序对数量为0,1,2,如果其中一个排列有序,那么另一个最多是1,这样对于这一对来说就是最优解,让某个序列全部排序,那么每一对都是最优的情况,那么就是最小的时候。

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
int a[200010],b[200010];
int main()
{
	int t;
	scanf("%d",&t);
	while(t--)
	{
		int n;
		scanf("%d",&n);
		vector<pair<int,int>>p;
		for(int i=1;i<=n;i++) scanf("%d",&a[i]);
		for(int i=1;i<=n;i++) scanf("%d",&b[i]);
		for(int i=1;i<=n;i++) p.push_back({a[i],b[i]});
		sort(p.begin(),p.end());
		for(int i=0;i<n;i++)
		{
			int x=p[i].first,y=p[i].second;
			printf("%d ",x);
		}
		printf("\n");
		for(int i=0;i<n;i++)
		{
			int x=p[i].first,y=p[i].second;
			printf("%d ",y);
		}
		printf("\n");
	}
}

XOR-distance(Problem - C - Codeforces

题目大意:现有三个数a,b,r,我们需要在[0,r]中选一个数x,使得|(a^x)-(b^x)|最小,输出最小值。

思路:既然涉及到位运算,那么我们就从二进制的角度来看:

显然如果a,b在二进制的某一位相同,那么和x进行异或之后,还是相同的,所以没什么变化,那么我们要想最小化结果,肯定还是要从不同位入手,很明显如果一个为0,一个为1,那么如果是0作为被减数,那么就要从高位退1下来参与,所以我们肯定将最高位1在的那个数中二进制下为1,而另一个数二进制下为0的位置更改一下,另外第一个相异的位置不能改动,因为要从这里退位下来。那么问题就解决了。那么我们直接进行位运算即可,注意不要超过r。

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
#define int long long
signed main()
{
	int t;
	scanf("%lld",&t);
	while(t--)
	{
		int a,b,r;
		scanf("%lld%lld%lld",&a,&b,&r);
		if(a<b) swap(a,b);
		int flag=1;
		for(int i=60;i>=0;i--)
		{
			int ta=(a>>i)&1,tb=(b>>i)&1;
			if(flag&&ta!=tb) 
			{
				flag=0;
				continue;
			}
			if(!flag)
			{
				if(ta==1&&tb==0) 
				{
					if((1ll<<i)<=r) 
					{
						r -= (1ll<<i);
						a ^= (1ll<<i);
						b ^= (1ll<<i);
					}
				}
			}
		}
		cout<<abs(a-b)<<endl;
	}
}

Blocking Elements(Problem - D - Codeforces

题目大意:现在有一个数组a[]和一个数组b[],我们要在a[]中将a[b[i]]的位置挑出来相加,剩下的以被挑走的位置为分隔,划出若干个区间,每个区间内部求和,这些和中的最大值作为结果,输出最小结果。举个例子:

思路:这里的结果肯定是在[1,sum]之内,所以想到二分,要快速求若干区间的和,那么想到预处理前缀和。现在最关键的问题就是二分的check函数怎么写。

我最初想的是就从头开始,一旦超过mid之后就以此划分,最后看挑出来的数是否小于mid,但是显然这个思路不可以。这么来看,我们第一次划分前的所有位置实际都可以作为第一次划分的位置,当然第二,第三次划分也是同理,那么就跟dfs联系起来了,对于搜每一次划分的位置。但是,显然时间复杂度太高了,这个题n的范围还有点大,那么就得换个思路。dfs简化就很容易想到动态规划,那么我们该如何动态规划呢?其实这里已经分析出了一个很重要的条件,每次划分的合法位置是一个区间,那么我们要想使结果最优,肯定是从这个区间中挑比较小的数加进结果。那么可以定义dp[i]表示以i位置为区间结尾,此时被挑选的数的和。因为我们在挑选的时候已经确定了每个区间的和是小于m的,进而来找划分位置,所以只要从这些合法的中找一个最小的即可。从一个区间中找最值,很容易想到滑动窗口,这里又是dp,那么就是单调队列优化dp的类型。还有一个问题,找合法区间的左端点时,是否必须从头开始找,显然不是,因为很容易可以发现,如果当前的i的左端点在j位置,那么后一个i的左端点一定是在j位置后面,因为我们确定区间只是通过区间和小于m来确定的。每一个值都是正数,那么区间和,区间一定不会再变长。所以我们只用看队列的开头是否需要弹出,来找合适的位置,因为队列是单增的,所以开头一定最小。

另外,结尾可能在任意位置,所以我们要循环判断一下。

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
#define int long long
int q[100010],a[100010],s[100010],dp[100010],hh,tt,n;
int check(int m)
{
	hh=0,tt=1;//数组模拟队列
	q[0]=0;
	for(int i=1;i<=n;i++)
	{
		while(hh<tt&&s[i-1]-s[q[hh]]>m) hh++;
		dp[i]=dp[q[hh]]+a[i];
		while(hh<tt&&dp[q[tt-1]]>=dp[i]) tt--;
		q[tt++]=i;
	}
	for(int i=1;i<=n;i++)
		if(dp[i]<=m&&s[n]-s[i]<=m) return 1;
	return 0;
}
signed main()
{
	int t;
	scanf("%lld",&t);
	while(t--)
	{
		scanf("%lld",&n);
		for(int i=1;i<=n;i++) scanf("%lld",&a[i]),s[i]=s[i-1]+a[i];
		int l=1,r=s[n];
		while(l<r)
		{
			int mid=(l+r)/2;
			if(check(mid)) r=mid;
			else l=mid+1;
		}
		printf("%lld\n",l);
	}
}

核心:虽然这道题既有区间和又有选点,但本质上考的是在限制区间的情况下求最小值,限制区间的条件比较隐蔽而已。 因为每一个划分的合法位置实际在一个区间中找,那么显然要找最小值,那么就是区间找最小值的问题,然后有多个位置,所以联系到单调队列优化的dp问题。

相关推荐
徐浪老师2 小时前
深入解析贪心算法及其应用实例
算法·贪心算法
软行2 小时前
LeetCode 单调栈 下一个更大元素 I
c语言·数据结构·算法·leetcode
钰爱&3 小时前
【操作系统】Linux之线程同步二(头歌作业)
linux·运维·算法
Ws_3 小时前
leetcode LCR 068 搜索插入位置
数据结构·python·算法·leetcode
灼华十一3 小时前
数据结构-布隆过滤器和可逆布隆过滤器
数据结构·算法·golang
adam_life4 小时前
OpenJudge_ 简单英文题_04:0/1 Knapsack
算法·动态规划
龙的爹23335 小时前
论文翻译 | The Capacity for Moral Self-Correction in Large Language Models
人工智能·深度学习·算法·机器学习·语言模型·自然语言处理·prompt
鸣弦artha6 小时前
蓝桥杯——杨辉三角
java·算法·蓝桥杯·eclipse
我是聪明的懒大王懒洋洋6 小时前
力扣力扣力:动态规划入门(1)
算法·leetcode·动态规划
丶Darling.6 小时前
Day44 | 动态规划 :状态机DP 买卖股票的最佳时机IV&&买卖股票的最佳时机III
算法·动态规划