【Codeforces Round 323 (Div 2)D】【暴力 脑洞 插入贡献思想】Once Again... 循环节重复T次后的LIS

D. Once Again...
time limit per test
1 second
memory limit per test
256 megabytes
input
standard input
output
standard output

You are given an array of positive integers a1, a2, ..., an × T of length n × T. We know that for any i > n it is true that ai = ai - n. Find the length of the longest non-decreasing sequence of the given array.

Input

The first line contains two space-separated integers: n, T (1 ≤ n ≤ 100, 1 ≤ T ≤ 107). The second line contains n space-separated integers a1, a2, ..., an (1 ≤ ai ≤ 300).

Output

Print a single number — the length of a sought sequence.

Sample test(s)
input
4 3
3 1 4 2
output
5
Note

The array given in the sample looks like that: 3, 1, 4, 2, 3, 1, 4, 2, 3, 1, 4, 2. The elements in bold form the largest non-decreasing subsequence.


#include<stdio.h>
#include<string.h>
#include<ctype.h>
#include<math.h>
#include<iostream>
#include<string>
#include<set>
#include<map>
#include<vector>
#include<queue>
#include<bitset>
#include<algorithm>
#include<time.h>
using namespace std;
void fre(){freopen("c://test//input.in","r",stdin);freopen("c://test//output.out","w",stdout);}
#define MS(x,y) memset(x,y,sizeof(x))
#define MC(x,y) memcpy(x,y,sizeof(x))
#define MP(x,y) make_pair(x,y)
#define ls o<<1
#define rs o<<1|1
typedef long long LL;
typedef unsigned long long UL;
typedef unsigned int UI;
template <class T1,class T2>inline void gmax(T1 &a,T2 b){if(b>a)a=b;}
template <class T1,class T2>inline void gmin(T1 &a,T2 b){if(b<a)a=b;}
const int N=1e4+1e3,M=0,Z=1e9+7,ms63=1061109567;
int casenum,casei;
int n,m,T;
int a[N],f[N];
int main()
{
	while(~scanf("%d%d",&n,&T))
	{
		int cir=min(100,T);
		int len=cir*n;
		for(int i=1;i<=n;i++)scanf("%d",&a[i]);
		int ans=0;
		for(int i=1;i<=len;i++)
		{
			if(i>n)a[i]=a[i-n];
			f[i]=1;
			int st=max(1,i-n);
			for(int j=st;j<i;j++)if(a[j]<=a[i])gmax(f[i],f[j]+1);
			gmax(ans,f[i]);
		}
		if(T>cir)
		{
			int now=0;
			for(int i=len+1;i<=len+n;i++)
			{
				a[i]=a[i-n];
				f[i]=1;
				for(int j=i-n;j<i;j++)if(a[j]<=a[i])gmax(f[i],f[j]+1);
				gmax(now,f[i]);
			}
			int dif=now-ans;
			ans+=(T-cir)*dif;
		}
		printf("%d\n",ans);
	}
	return 0;
}
/*
【trick&&吐槽】
1,形成插入思想。用暴力思维解决问题。
2,注意细节。

【题意】
给你一个长度为n([1,100])的数列,这个数列形成了T次,即总共有n*T个数出现。
对于位置i(i>n),有a[i]=a[i-n]。
让你输出数列a[]的最长单调不下降子序列的长度

【类型】
脑洞

【分析】
这题我观察到,因为数的个数最多才只有n个,所以就算数列重复的次数很大,形成的递增的数的个数也不过只有n。
于是我们会有大量相等的LIS元素出现。

基于这个观察到的性质,我一开始选择的做法是,向前枚举一个循环节,向后枚举一个循环节,两个循环节求出LIS,然后枚举中间字符为相同,
三部分拼接起来更新答案。

但是这种做法是错误的。
比如数据9 10 11 12 5 6 7 8 1 2 3 4
于是我们发现,形成递增的循环节数量不止只有最前一个最后一个,而可能是很多个。
于是这里要怎么办?

方法一,枚举前循环节数量,枚举后循环节数量,然后再做拼接。
然而这个方法的时间复杂度巨大,实现起来细节众多,于是比赛的时候我就误入歧途,连这么水的一道题都没有做出来TwT

方法二,抓主要矛盾!
这道题的n只有100,而循环节的数量可达1e7.
如果循环节的数量只有100,那么我们甚至可以直接求LIS,
于是问题转变为两部分——
1,T<=100,这时可以直接暴力
2,T>100时,我们再新增一个循环节,这个循环节对之前LIS的影响,一定是这个循环节以平的形式插入到之前的LIS数列中。
于是我们求出这时插入新增循环的贡献量,而之后每次插入新增循环节,插入新增数量*单一贡献就是对答案的贡献。由此战胜答案即可。

至于LIS,我们用O((100n)^2)求也可,用O((100n)log(100n))也可。

【时间复杂度&&优化】
O((100n)^2)
O((100n)log(100n))

【数据】
12 3
9 10 11 12 5 6 7 8 1 2 3 4 

*/

你可能感兴趣的:(codeforces,暴力,脑洞)