题目要求

总时间限制: 2000ms 内存限制: 65536kB

描述

一个数的序列bi,当b1 < b2 < … < bS的时候,我们称这个序列是上升的。
对于给定的一个序列(a1, a2, …, aN),我们可以得到一些上升的子序列(ai1, ai2, …, aiK),这里1 <= i1 < i2 < … < iK <= N。
比如,对于序列(1, 7, 3, 5, 9, 4, 8),有它的一些上升子序列,如(1, 7), (3, 4, 8)等等。
这些子序列中最长的长度是4,比如子序列(1, 3, 5, 8)。
你的任务,就是对于给定的序列,求出最长上升子序列的长度。

输入

输入的第一行是序列的长度N (1 <= N <= 1000)。
第二行给出序列中的N个整数,这些整数的取值范围都在0到10000。

输出

最长上升子序列的长度。

样例输入

7
1 7 3 5 9 4 8

样例输出

4

来源

翻译自 Northeastern Europe 2002, Far-Eastern Subregion 的比赛试题

解题思路

1.找子问题
“求序列的前n个元素的最长上升子序列的长度”是个子问题,但是这样分解子问题,不具有“无后效性”:假设F(n)=x,但可能有多个序列满足F(n)=x。有的序列的最后一个元素比an+1小,则加上an+1就能形成更长上升子序列;有的序列最后一个元素不比an+1小……以后的事情受如何达到状态n的影响,不符合“无后效性”。

“求以ak(k=1, 2, 3…N)为终点的最长上升子序列的长度”一个上升子序列中最右边的那个数,称为该子序列的“终点”。虽然这个子问题和原问题形式上并不完全一样,但是只要这N个子问题都解决了,那么这N个子问题的解中,最大的那个就是整个问题的。

2.确定状态
子问题只和一个变量-- 数字的位置相关。因此序列中数的位置k 就是“状态”,而状态 k 对应的“值”,就是以ak做为“终点”的最长上升子序列的长度状态一共有N。

3.找出状态转移方程
maxLen (k)表示以ak做为“终点”的最长上升子序列的长度那么:
初始状态:maxLen (1) = 1 。
maxLen (k) = max { maxLen (i):1<=i < k 且 ai < ak且 k≠1 } + 1
若找不到这样的i,则maxLen(k) = 1。
maxLen(k)的值,就是在ak左边,“终点”数值小于ak ,且长度最大的那个上升子序列的长度再加1。因为ak左边任何“终点”小于ak的子序列,加上ak后就能形成一个更长的上升子序。

代码

#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std;
const int MAXN =1010;
int a[MAXN]; int maxLen[MAXN];
int main()
{int N; cin >> N;for( int i = 1;i <= N;++i){cin >> a[i]; maxLen[i] = 1;}for( int i = 2; i <= N; ++i){//每次求以第i个数为终点的最长上升子序列的长度for( int j = 1; j < i; ++j)//察看以第j个数为终点的最长上升子序列if( a[i] > a[j] )maxLen[i] = max(maxLen[i],maxLen[j]+1);}cout << * max_element(maxLen+1,maxLen + N + 1 );return 0;
} //时间复杂度O(N^2)

百练2757:最长上升子序列相关推荐

  1. OpenJudge 2757 最长上升子序列 / Poj 2533 Longest Ordered Subsequence

    1.链接地址: http://poj.org/problem?id=2533 http://bailian.openjudge.cn/practice/2757 2.题目: 总Time Limit: ...

  2. 北大OJ百练——3179:最长单词(C语言)

    废话不多说,先来上题目: OJ的这题关键在于如何计算我所选择的开始位置和结束位置.我用的方法是有4个参数来存储起来,start, buffStart, lenth, maxLenth. start: ...

  3. 北大OJ百练——4073:最长公共字符串后缀(C语言)

    刚刚看到一道北大的OJ题,很简单的一道题.原题如下(偷个懒,直接截图): 看完这道题,我想大家都和我一样觉得这道题很简单,事实也是如此,毕竟通过率很高. 我先来说一下我的思路吧.我是想先把这些所有的字 ...

  4. Bailian2757 最长上升子序列【DP】

    2757:最长上升子序列 描述 一个数的序列bi,当b1 < b2 < - < bS的时候,我们称这个序列是上升的.对于给定的一个序列(a1, a2, -, aN),我们可以得到一些 ...

  5. 百练(十三~十六)题解

    百练(十三) Bailian2806 公共子序列[最长公共子序列+DP] - 海岛Blog - CSDN博客 Bailian3143 验证"歌德巴赫猜想"[筛选法]_海岛Blog- ...

  6. 百练(九~十二)题解

    百练(九) Bailian2801 填词 POJ1629 ZOJ1546 Fillword[排序] - 海岛Blog - CSDN博客 POJ1088 Bailian1088 滑雪[DFS+记忆化搜索 ...

  7. 程序设计入门经典题解(百练篇)

    参考链接:PKU百练题解(Bailian) Bailian1017 装箱问题[贪心] - 海岛Blog - CSDN博客 POJ1088 Bailian1088 滑雪[DFS+记忆化搜索]_海岛Blo ...

  8. 北大培训课动态规划----神奇的口袋(百练2755)

    北京大学暑期课<ACM/ICPC竞赛训练> ppt摘取 什么是动态规划? ●递归到动规的一般转化方法  递归函数有n个参数,就定义一个n维的数组,数组 的下标是递归函数参数的取值范围,数组 ...

  9. 动态规划——最长上升子序列问题 两种角度及优化算法

    最长上升子序列 OpenJ_Bailian - 2757 一个数的序列 bi,当 b1 < b2 < ... < bS的时候,我们称这个序列是上升的.对于给定的一个序列( a1, a ...

最新文章

  1. antd Drawer 如何实现自动刷新
  2. 让迅雷的胃口变得更大!
  3. 世界是一台超级计算机,这个世界其实是一个超级计算机
  4. 基于DCT系数的实时监控中运动目标检测
  5. Python风格总结:Python3 标准库概览
  6. 从浏览器输入地址到渲染出网页这个过程发生了什么?
  7. 使用SQL Coalesce函数查询数据
  8. Cfs三层靶机内网渗透模拟
  9. 从零打造Android计算器(安卓开发初体验)
  10. 2019年末逆向复习系列之从猫眼字体反爬分析谈谈字体反爬的前世今生
  11. unix下的softlink和hardlink
  12. redis为什么这么快
  13. fiddler抓包如何只抓手机端的包 不抓电脑的包
  14. ch2_8_2求解幸运数问题
  15. pandas中drop用法_机器学习笔记:Pandas的delete、drop函数的用法
  16. 关于APIC的一些概念
  17. Python爬取微信公众号文章、点赞数
  18. 数据结构期末考试——选择题
  19. 天馈线测试仪具备什么功能
  20. 利用Composer搭建企业内部仓库

热门文章

  1. Action类中通过ServletActionContext来获取web资源
  2. [SinGuLaRiTy] KM算法
  3. Integration testing
  4. 【转】iOS开发6:UIActionSheet与UIAlertView
  5. 在nagios中使用nrpe自定义脚本
  6. android activity 回调函数,Android Activity的生命周期
  7. 6-3 求链表的倒数第m个元素
  8. 每日程序C语言8-打印“水仙花数”
  9. JAVA 海啸_java线程总结
  10. SSM-网站后台管理系统制作(2)---SSM基本工作原理