hdu 4604 Deque(最长上升与下降子序列-能够重复)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4604
这个题解有点问题,暂时没时间改,还是参考别人的吧
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <iostream>
#include <cstring>
#include <queue>
#include <vector>
#define maxn 100500
using namespace std; const int INF = 0x3f3f3f; int a[maxn];
int s1[maxn],s2[maxn];
int ans;
int n; int search_lower_bound(int l,int h,int m){
if(l == h) return l;
int mid = (l + h)/;
if(s1[mid] <= m) return search_lower_bound(mid+,h,m);
else return search_lower_bound(l,mid,m);
}
int search_upper_bound(int l,int h,int m){
if(l == h) return l;
int mid = (l + h)/;
if(s2[mid] >= m) return search_upper_bound(mid+,h,m);
else return search_upper_bound(l,mid,m);
} int main()
{
// if(freopen("input.txt","r",stdin)== NULL) {printf("Error\n"); exit(0);} int T;
cin>>T;
while(T--){
cin>>n;
for(int i=;i<n;i++) scanf("%d",&a[i]);
for(int i=;i<=n/;i++){
int temp;
temp = a[n-i-];
a[n-i-] = a[i];
a[i] = temp;
}
s1[] = s2[] = a[];
int rear1 = ,rear2 = ;
ans = ;
for(int i=;i<n;i++){
int temp1,temp2; if(s1[rear1] < a[i]) temp1 = ++rear1;
else{
temp1 = search_lower_bound(,rear1,a[i]);
if(s1[temp1] == a[i]){
for(int i=++rear1;i>temp1;i--) s1[i] = s1[i-];
temp1++; // printf("&&&&&& %d\n",rear1);
}
}
s1[temp1] = a[i]; if(s2[rear2] > a[i]) temp2 = ++rear2;
else{
temp2 = search_upper_bound(,rear2,a[i]);
/*if(s2[temp2] == a[i]){
for(int i=++rear2;i>temp2;i--) s2[i] = s2[i-1];
temp2++; printf("****** %d\n",rear2);
}*/
}
s2[temp2] = a[i];
ans = max(ans,temp1+temp2+);
}
cout<<ans<<endl;
}
}
这题虽然过了,但存在一个问题。是数据弱了点。正确的思路:
(引用)
考虑题目的一个简化版本:使双端队列单调上升。对于序列A和队列Q,找到队列中最早出现的数字Ax,则Ax将Q分成的两个部分分别是原序列中以Ax开始的最长上升和最长下降序列,答案即为这两者之和的最大值。而对于本题,由于存在相同元素,所以只要找到以Ax 为起点的最长不下降序列和最长不上升序列的和,然后减去两个里面出现Ax次数的最小值即可。
我忽略了下降中的元素可重复,应该要考虑,在减去两个里面都存在的Ax重复次数;
所以有了大神的思路写出全新的代码:
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <iostream>
#include <cstring>
#include <queue>
#include <vector>
#define maxn 105000
using namespace std; const int INF = 0x3f3f3f; int d1[maxn],d3[maxn];
int g1[maxn],g2[maxn];
int a[maxn],b[maxn];
int n; int main()
{
//if(freopen("input.txt","r",stdin)== NULL) {printf("Error\n"); exit(0);} int T;
cin>>T;
while(T--){
cin>>n;
for(int i=n-;i>=;i--){
scanf("%d",&a[i]);
b[i] = -a[i];
}
for(int i=;i<=n;i++){
g1[i] = INF;
g2[i] = INF;
}
int ans = ;
for(int i=;i<n;i++){
int k1 = upper_bound(g1+,g1+n+,a[i]) - g1;
int k2 = upper_bound(g2+,g2+n+,b[i]) - g2;
int k3 = lower_bound(g2+,g2+n+,b[i]) - g2;
g1[k1] = a[i];
g2[k2] = b[i];
d1[i] = k1;
d3[i] = k3;
ans = max(ans,d1[i] + d3[i] - );
}
printf("%d\n",ans);
}
}
这才是正确的方法,太精妙了。
lower_bound()是binary_search()的特殊形式. 此函数搜索给定的序列[first, end)中val可插入的第一个位置; 或者说, 它返回序列中遇到的第一个不小于val的元素的迭代器, 如果所有元素都小于val则返回“end”. 从函数要求待搜索序列是有序的.
lower_bound()的返回值乃是一个指向val可以安全插入的位置的迭代器. 在比较函数f被指定之前, 默认使用<操作符进行排序.
而upper_bound()是则是允许重复;
求最长下降就把a[] 变为相反数 b[],求b[]的最长上升子序列;
hdu 4604 Deque(最长上升与下降子序列-能够重复)的更多相关文章
- HDU 4604 deque 最长上升子序列
枚举每个位置,求以num[i]为起点的最长不下降子序列和以num[i]为结尾的最长不递增子序列. 并且把相同值的个数统计一下,最后要减去算重复了的. 比如: 1 9 4 4 2 2 2 3 3 3 7 ...
- HDU 4604 Deque(最长上升子序)
题目链接 本来就对N*log(N)算法不大会....然后各种跪了,求出最长不下降+最长不上升-最少相同元素.求相同元素,用二分求上界搞的.代码里4个二分.... #include <cstdio ...
- 最长非降/下降子序列问题(DP)(待续...)
注意:抽象成以下描述即为最长非降/下降子序列问题(一维状态) 问题描述:在一个无序的序列a1,a2,a3,a4…an里,找到一个最长的序列满足:(不要求连续) ai<=aj<=ak…< ...
- hdu 4604 Deque(最长不下降子序列)
从后向前对已搜点做两遍LIS(最长不下降子序列),分别求出已搜点的最长递增.递减子序列长度.这样一直搜到第一个点,就得到了整个序列的最长递增.递减子序列的长度,即最长递减子序列在前,最长递增子序列在后 ...
- HDU 4604 Deque 最长子序列
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4604 Deque Time Limit: 4000/2000 MS (Java/Others) ...
- HDU 4604 Deque 二分最长上升子序列
题目大意就是给一个deque 然后有n个数,依次进行操作,每种操作,你可以把这个数放在deque首部,也可以放在尾部,也可以扔掉不管,但是要保证deque中的数是非递减的.最要求deque中最长能是多 ...
- hdu 4604 Deque
http://acm.hdu.edu.cn/showproblem.php?pid=4604 将原数组根据其大小关系处理后 使其大小全在10^5内 处理后为 a1,a2,a3.....an 最优deq ...
- Longest Ordered Subsequence POJ - 2533 dp 最长上升/不下降 子序列
#include<iostream> using namespace std ; ; int f[N]; int a[N]; int n; int main() { cin>> ...
- HDU 6357.Hills And Valleys-字符串非严格递增子序列(LIS最长非下降子序列)+动态规划(区间翻转l,r找最长非递减子序列),好题哇 (2018 Multi-University Training Contest 5 1008)
6357. Hills And Valleys 自己感觉这是个好题,应该是经典题目,所以半路选手补了这道字符串的动态规划题目. 题意就是给你一个串,翻转任意区间一次,求最长的非下降子序列. 一看题面写 ...
随机推荐
- SQL存储过程基于字段名传入的字符串拼接.
--定义存储过程. Create PROCEDURE Usp_Static ), ), --分组字段. ), --统计字段. ), --表头字段. ) --聚会函数. AS ) --存储游标执行的列. ...
- angular调用WCF服务,读取文件夹下图片显示列表,下载另存为图片
读取文件夹下的文件 public string ReadImagesPaths() { string result = string.Empty; try { string path = System ...
- svn的初级使用
首先呢 你需要下载一个软件 比如说是 Cornerstone. 进行安装好之后 然后 然后输入账号密码 就可以了 然后去xcode去进行相关的配置 点击第二个进入 偏好设置 点击最下边的+ 点击第二 ...
- String 方法
import java.lang.*; /** * 1.查找"asdfjvjadsffvaadfkfasaffdsasdffadsafafsafdadsfaafd" * 该字符串中 ...
- 自动化工具word文档批量转html
企业有很多的科室,科室的每个人或多或少都会写一些文档,有些文档领导需要浏览,解决的办法是将编辑的文档打印出来,供领导浏览,或是为了节约企业成本,文档就在人与人这间或部门之间copy过来,copy过去. ...
- jquery 判断页面滚动到底部
$(document).scrollTop() 获取垂直滚动的距离 即当前滚动的地方的窗口顶端到整个页面顶端的距离$(document).scrollLeft() 这是获取水平滚动条的距离获取顶端 只 ...
- Mac下搭建php开发环境
Mac OS X 内置了Apache 和 PHP,这样使用起来非常方便.本文以Mac OS X 10.6.3为例.主要内容包括: 启动Apache 运行PHP 安装MySQL 使用phpMyAdmin ...
- bzoj1311: 最优压缩
Description 其中: Auv是与Aij相邻的像素(为了简化,认为(i-1,j),(i+1,j,(i,j-1),(i,j+1)为相邻元素); Wij取值0或者1,表示Aij修改后取V0或者V ...
- iOS应用开发:什么是ARC?
iOS应用开发:什么是ARC? 博客分类: Phone / IOS / Objective-C / Swift ARC是什么 ARC是iOS 5推出的新功能,全称叫 ARC(Automatic R ...
- iOS的launch image --备用
当我们打开一款应用程序的时候,首先映入眼帘的往往并不是程序的主界面,而是经过精心设计的欢迎界面,这个界面通常会停留几秒钟,然后消失.看似很平常的一个小小的欢迎界面,其实还大有讲究. 一.为什么会出现欢 ...