C. Colorful Table CodeTON Round 6 (Div. 1 + Div. 2, Rated, Prizes!)

题目大意:有一个n个数的数组a,每个数大小不超过k,构建一个n*n的方阵b,其中每个元素b[i,j]=min(a[i],a[j]),求对于1~k的每一个数x,包含矩阵中所有x的最小的矩形的宽+长的值

1<=n,k<=1e5

思路:通过观察可以发现,这个方阵是关于左上-右下的直线对称的,所以我们要找的矩形都是正方形,因为方阵中的值是取最小值得来的,所以每个数构成的正方形的边界就是a中最左边和最右边大于该数的位置,那么问题就转化成了在数组中求每个数最左和最右边大于这个数的位置之间的长度。

首先考虑每个数的左边界怎么求,因为每个数的边界都是最靠左的那个,所以从左向右遍历,我们每遇到一个没处理过的数,所有小于等于这个数的数的左边界都是这个位置,然后这些数都被标记为已处理。

右边界也依照此法从右向左遍历,这样我们就得到了每个数构成的正方形的边长为右边界-左边界+1,再乘二就是答案。

cpp 复制代码
//#include<__msvc_all_public_headers.hpp>
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 1e5 + 5;
ll n;
ll a[N];
int l[N], r[N];
bool vis[N];
int k;
void init()
{
	for (int i = 1; i <= k; i++)
	{
		vis[i] = 0;
	}
}
void solve()
{
	cin >> n >> k;
	init();
	for (int i = 1; i <= n; i++)
	{
		cin >> a[i];
		vis[a[i]] = 1;//记录一下哪些数在数组中出现过
	}
	int now = 1;
	for (int i = 1; i <= n; i++)
	{		
		while (now <= a[i])
		{
			l[now] = i;//更新所有小于等于这个数的数的左边界
			now++;
		}
	}
	now = 1;
	for (int i = n; i >= 1; i--)
	{
		while (now <= a[i])
		{
			r[now] = i;
			now++;
		}
	}
	for (int i = 1; i <= k; i++)
	{
		if (!vis[i])
		{//没出现的数就是0
			cout << 0 << " ";
			continue;
		}
		cout << (r[i] - l[i] + 1) * 2 << " ";
	}
	cout << endl;
}
int main()
{
	ios::sync_with_stdio(false);
	cin.tie(0);
	int t;
	cin>>t;
	while (t--)
	{
		solve();
	}
	return 0;
}



 
相关推荐
_日拱一卒1 分钟前
LeetCode:除了自身以外数组的乘积
数据结构·算法·leetcode
计算机安禾6 分钟前
【数据结构与算法】第36篇:排序大总结:稳定性、时间复杂度与适用场景
c语言·数据结构·c++·算法·链表·线性回归·visual studio
unicrom_深圳市由你创科技10 分钟前
做虚拟示波器这种实时波形显示的上位机,用什么语言?
c++·python·c#
SatVision炼金士12 分钟前
合成孔径雷达干涉测量(InSAR)沉降监测算法体系
算法
wuweijianlove16 分钟前
算法稳定性与数值误差传播研究的技术2
算法
无限进步_16 分钟前
【C++】电话号码的字母组合:从有限处理到通用解法
开发语言·c++·ide·windows·git·github·visual studio
计算机安禾38 分钟前
【数据结构与算法】第35篇:归并排序与基数排序
c语言·数据结构·vscode·算法·排序算法·哈希算法·visual studio
专注API从业者42 分钟前
淘宝商品详情 API 与爬虫技术的边界:合法接入与反爬策略的技术博弈
大数据·数据结构·数据库·爬虫
C++ 老炮儿的技术栈1 小时前
GCC编译时无法向/tmp 目录写入临时汇编文件,因为设备空间不足,解决
linux·运维·开发语言·汇编·c++·git·qt
爱码小白1 小时前
MySQL 单表查询练习题汇总
数据库·python·算法