最长上升子序列(9)

1 问题

(郭炜《算法基础于在线实践》)

1.1 问题描述

给定一个序列(a1,a2,a3,...,an),从中任意提取一个子序列(bk1,bk2,bk3,...,bkm),若满足bk1 < bk2 < ... <bkm,则b是a的一个上升子序列。求a的最长上升子序列的长度。

1.2 输入数据

第一行N是序列的长度,第二行有N个整数,是序列的各个元素。

1.3 输入样例

1 7 3 5 9 4 8

1.4 输出样例

4    //1 3 5 8 或1 3 4 8

2 问题分析

问题分解,求序列前N项的最长上升子序列,等于前N-1项和第N项进行判断,但是无后效性,反例:序列(1 7 3 5 4 5),N = 6,前6项为前5项和第6项,前5项的最长上升子序列可以为(1 3 5)或者(1 3 4),因为最后一个数有多种状态,直接影响第6项的结果,即(1 3 5)或者(1 3 4 5),这样就会导致结果的不同,不能用这种子问题分解方法。

换个思路,一个上升子序列,将最右边的那个元素定义为终点,子问题分解时,F(n)定义为以第n个元素为终点(不是前n项)的上升子序列,则F(n+1)可以为F(n)和第n+1个元素对比来判断,若第n+1个元素 > 第n个元素,则F(n+1) = max(F(n))+1,F(n+1)就为前面所有的最长上升子序列的长度+1。若n=1,则结果为1,递归终止条件。

设最大长度为maxlen(n),maxlen(1) = 1,由maxlen(1)和A[2]可以得到maxlen(2),由maxlen(2)、maxlen(1)和A[3]就可以得到maxlen(3),以此类推,不需要递归就可以得到答案。 

3 代码编写

#include <iostream>
#include <algorithm>
using namespace std;

void test() {
	int N;
	cin >> N;
	int* A = new int[N + 1];
	int* maxlen = new int[N + 1];
	A[0] = 0; maxlen[0] = 1;
	for (int i = 1; i <= N; i++) {
		cin >> A[i];
		maxlen[i] = 1;
	}

	for (int i = 2; i <= N; i++) {
		for (int j = 1; j < i; j++) {
			if (A[i] > A[j])
				maxlen[i] = max(maxlen[i], maxlen[j] + 1);
		}
	}

	int temp = 0;
	for (int i = 1; i <= N; i++) {
		if (temp < maxlen[i]) {
			temp = maxlen[i];
		}
	}
	cout << temp << endl;
}

int main()
{
	test();
	return 0;
}

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值