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

本文主要是介绍【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: nT (1 ≤ n ≤ 1001 ≤ 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, 23, 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 Round 323 (Div 2)D】【暴力 脑洞 插入贡献思想】Once Again... 循环节重复T次后的LIS的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/610497

相关文章

Oracle数据库使用 listagg去重删除重复数据的方法汇总

《Oracle数据库使用listagg去重删除重复数据的方法汇总》文章介绍了在Oracle数据库中使用LISTAGG和XMLAGG函数进行字符串聚合并去重的方法,包括去重聚合、使用XML解析和CLO... 目录案例表第一种:使用wm_concat() + distinct去重聚合第二种:使用listagg,

MySQL中删除重复数据SQL的三种写法

《MySQL中删除重复数据SQL的三种写法》:本文主要介绍MySQL中删除重复数据SQL的三种写法,文中通过代码示例讲解的非常详细,对大家的学习或工作有一定的帮助,需要的朋友可以参考下... 目录方法一:使用 left join + 子查询删除重复数据(推荐)方法二:创建临时表(需分多步执行,逻辑清晰,但会

JAVA中while循环的使用与注意事项

《JAVA中while循环的使用与注意事项》:本文主要介绍while循环在编程中的应用,包括其基本结构、语句示例、适用场景以及注意事项,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录while循环1. 什么是while循环2. while循环的语句3.while循环的适用场景以及优势4. 注意

Python中的异步:async 和 await以及操作中的事件循环、回调和异常

《Python中的异步:async和await以及操作中的事件循环、回调和异常》在现代编程中,异步操作在处理I/O密集型任务时,可以显著提高程序的性能和响应速度,Python提供了asyn... 目录引言什么是异步操作?python 中的异步编程基础async 和 await 关键字asyncio 模块理论

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1496(用hash思想统计数目)

作为一个刚学hash的孩子,感觉这道题目很不错,灵活的运用的数组的下标。 解题步骤:如果用常规方法解,那么时间复杂度为O(n^4),肯定会超时,然后参考了网上的解题方法,将等式分成两个部分,a*x1^2+b*x2^2和c*x3^2+d*x4^2, 各自作为数组的下标,如果两部分相加为0,则满足等式; 代码如下: #include<iostream>#include<algorithm

poj2406(连续重复子串)

题意:判断串s是不是str^n,求str的最大长度。 解题思路:kmp可解,后缀数组的倍增算法超时。next[i]表示在第i位匹配失败后,自动跳转到next[i],所以1到next[n]这个串 等于 n-next[n]+1到n这个串。 代码如下; #include<iostream>#include<algorithm>#include<stdio.h>#include<math.

poj3261(可重复k次的最长子串)

题意:可重复k次的最长子串 解题思路:求所有区间[x,x+k-1]中的最小值的最大值。求sa时间复杂度Nlog(N),求最值时间复杂度N*N,但实际复杂度很低。题目数据也比较水,不然估计过不了。 代码入下: #include<iostream>#include<algorithm>#include<stdio.h>#include<math.h>#include<cstring

Codeforces Round #240 (Div. 2) E分治算法探究1

Codeforces Round #240 (Div. 2) E  http://codeforces.com/contest/415/problem/E 2^n个数,每次操作将其分成2^q份,对于每一份内部的数进行翻转(逆序),每次操作完后输出操作后新序列的逆序对数。 图一:  划分子问题。 图二: 分而治之,=>  合并 。 图三: 回溯:

Codeforces Round #261 (Div. 2)小记

A  XX注意最后输出满足条件,我也不知道为什么写的这么长。 #define X first#define Y secondvector<pair<int , int> > a ;int can(pair<int , int> c){return -1000 <= c.X && c.X <= 1000&& -1000 <= c.Y && c.Y <= 1000 ;}int m