首页 > 其他分享 >UVA1471 防线 Defense Lines 题解

UVA1471 防线 Defense Lines 题解

时间:2023-10-02 17:45:40浏览次数:48  
标签:ch 题解 Lines long UVA1471 数组 序列 dpl define

传送门


首先可以将题意大概可以简化为:取两端不重复的连续子序列,组成一个最长的连续递增子序列。

我们先 dp 预处理出以 \(i\) 为结尾的连续递增子序列长度 \(dpr_{i}\)。

同样预处理出以 \(i\) 为开头的连续递增子序列长度 \(dpl_{i}\)。

考虑对于每个 \(dpr_{i}\),找到满足 \(a_{j}<a_{i}\) 的最大的 \(dpl_{i}\)。

这个过程可以建一个桶,用二分查找来优化,但是问题来了:\(a_{i}\leq 10^9\),用 \(a_{i}\) 当数组下标显然不行。

这里有一个常见的小技巧:将数值和下标互换,以 \(dpl_{i}\) 作为数组下标,最小的 \(a_{i}\) 作为数组存的值。

所以我们开一个 \(t\) 数组,\(t_{i}=j\) 表示长度为 \(i\) 的序列的结尾最小值为 \(j\)。对于每个 \(dpr_{i}\),在这个数组中二分查找即可。

时间复杂度:\(\mathcal{O}(Tn\log{n})\)。


\(\mathcal{Code}\):

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define ull unsigned long long
#define db double
#define il inline
#define re register
#define INF 0x3f3f3f3f
#define LINF 0x3f3f3f3f3f3f3f3f
#define F(i,a,b) for(re int (i)=(a);(i)<=(b);(i)++)
#define DF(i,a,b) for(re int (i)=(a);(i)>=(b);(i)--)
#define G(i,u) for(re int (i)=head[u];(i);(i)=nxt[(i)])
inline ll read(){ll x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-48;ch=getchar();}return x*f;}
const int N=200010;
int n;
int a[N],dpl[N],dpr[N],t[N];
int main()
{
	int T=read();
	while(T--)
	{
		memset(t,0x3f,sizeof(t));
		n=read();
		dpl[0]=dpl[n+1]=dpr[0]=dpr[n+1]=0;
		a[0]=INF,a[n+1]=-INF;
		F(i,1,n) a[i]=read();
		F(i,1,n)
		{
			if(a[i]>a[i-1]) dpl[i]=dpl[i-1]+1;
			else dpl[i]=1;
		}
		DF(i,n,1)
		{
			if(a[i]<a[i+1]) dpr[i]=dpr[i+1]+1;
			else dpr[i]=1;
		}
		int ans=0;
		F(i,1,n)
		{
			t[dpl[i]]=min(t[dpl[i]],a[i]);
			int k=lower_bound(t+1,t+n+1,a[i])-t-1;
			ans=max(ans,k+dpr[i]);
		}
		printf("%d\n",ans);
	}
	return 0;
}

标签:ch,题解,Lines,long,UVA1471,数组,序列,dpl,define
From: https://www.cnblogs.com/MooSheng/p/17740260.html

相关文章

  • P5503 灯塔 题解
    决策单调性二分传送门数据加强版:P3515前置知识:二分,决策单调性首先很容易写出答案式子:\[ans_{i}=\max_{j=i}^{n}{(a_{j}-a_{i}+\lceil\sqrt{\left|i-j\right|}\rceil)}\]先将向上取整符号拆掉,只要在输出时处理就行。再将绝对值符号拆掉,分\(j<i\)和\(j>i\)的情况......
  • 题解 Codeforces Round 901 (Div. 1) / CF1874A~E
    题解CodeforcesRound901(Div.1)/CF1874A~E比赛情况:过了AB。赛后发现B是假复杂度。https://codeforc.es/contest/1874A.JellyfishandGameProblemAlice&Bob又在博弈,Alice手上有\(n\)个苹果,第\(i\)个苹果的价值是\(a_i\);Bob手上有\(m\)个苹果,第\(i\)......
  • [题解]AT_abc240_f [ABC240F] Sum Sum Max
    思路题目要求的是\(\max_{a=1}^{n}\{\sum_{i=1}^{a}\sum_{j=1}^{a}{A_j}\}\),所以我们将\(\sum_{i=1}^{a}\sum_{j=1}^{a}{A_j}\)化简一下,得:\[i\timesA_1+(i-1)\timesA_2+\dots+1\timesA_x\]在\(a\)每增加\(1\)时,这个和\(s\)将会变为\(s+......
  • [题解]AT_abc245_f [ABC245F] Endless Walk
    思路首先我们可以发现,在任意一个节点数量大于\(1\)的强连通分量中的点都满足条件。所以,我们可以对这张图跑一边TarJan。但是这样是错的,因为我们还需要考虑节点数量为\(1\)的强连通分量。如果这种连通分量能够到达任意一个节点数量大于\(1\)的强连通分量,那么,这个连通分......
  • CSES.1141 C++题解
    题意传送门有一个长度为\(n\)的歌单,问最长多少首歌互不相同?每首歌用一个\(1-10^9\)的整数表示。样例输入812132742样例输出5算法双指针算法。桶思想。对于歌单中重复出现的数,可以用桶来存储。定义两个指针i,j,i指向大数,j指向小数。当出现某个桶的数大于1时,则......
  • CF1878C Vasilije in Cacak 题解
    题目传送门简化题意有\(t\)组询问,每次询问是否能从\(1\simn\)中选择\(k\)个数使得它们的和为\(x\)。解法考虑临界情况,从\(1\simn\)中选择最小的\(k\)个数时和为\(\sum\limits_{i=1}^ki=\dfrac{(k+1)k}{2}\),从\(1\simn\)中选择最大的\(k\)个数时和为\(......
  • Codeforces 1765H 题解
    题目大意题目大意给定一个\(n\)个点和\(m\)条边的有向图,并给定\(p_1,p_2,\cdots,p_n\)表示第\(i\)个点的拓扑序必须小于等于\(p_i\),求出每个点的最小拓扑序。题解题解题目要求拓扑序尽量小,转换一下就是在反图上拓扑序尽量大。考虑拓扑排序,当一个点不得不入队......
  • UVA10054 The Necklace 题解
    好可恶一道题,怎么没人告诉我输出之间有空行(思路是先抽象成图,然后跑一边dfs记录边的前后顺序。对于不能成环的情况,只需要再开个数组记录度数判断奇点即可。若存在奇点则break掉,剩下的跑dfs、//producedbymiya555//stupidmistakes:1.多测要清空2.输出之间有空行//ideas:d......
  • 题解 hdu 1269 迷宫城堡
    找点图论练习题写,发现hdu又寄了,那就发到blog里吧。思路:tarjan缩点判断DAG中点数是否为1。若是,则该图为强连通图。 //producedbymiya555//stupidmistakes:多测记得清空//ideas:tarjan模板#include<bits/stdc++.h>usingnamespacestd;constintN=10010;intn,m,low[......
  • 题解 小 a 和 uim 之大逃离
    题目链接首先可以想到设状态\(k_1,k_2\)表示小\(a\)和小\(uim\)分别表示他们目前取得的得分,那么最终的答案便是\(k_1=k_2\)的时候。但是这样设置状态的复杂度无疑是高的。并且十分浪费,所以考虑设\(z\)表示\(k_1-k_2\)的值。那么\(z=0\)就是答案。接着考虑如何处......