设计一个O(n2)时间的算法,找出由n个数组成的序列的最长单调递增子序列。

题目如下:

设计一个O(n2)时间的算法,找出由n个数组成的序列的最长单调递增子序列。

解题思路:

思路一:简单dp,求最长递增子序列,即为求其与已经排好序的序列的公共子序列


/*设计一个O(n2)时间的算法,找出由n个数组成的序列的最长单调递增子序列。*/
#include <iostream>
#include<vector>
#include<algorithm>
using namespace std;
#define N 100

void fun(int n);

vector< int > a,b;

void main()
{
	int i,n,temp;
	printf("input n:");
	scanf("%d",&n);
	printf("input the array.\n");
	for(i=0;i<n&&cin>>temp;i++)
		a.push_back(temp);
	b=a;
	sort(b.begin(),b.end());
	fun(n);

}

void fun(int n)
{
	int m[N][N],flag[N];
	if(a[0]!=b[0])
		m[0][0]=0;
	else
		m[0][0]=1;
	for(int i=0;i<n;i++)
	{
		for(int j=0;j<n;j++)
		{
			if(i==0&&j==0)
			{
				continue;
			}
			if(a[i]==b[j])
			{
				if(i==0)
					m[i][j]=m[0][j-1]+1;
				else if(j==0)
					m[i][j]=m[i-1][0]+1;
				else
				    m[i][j]=m[i-1][j-1]+1;
			}
			else
			{
				if(i==0)
					m[i][j]=m[i][j-1];
				else if(j==0)
					m[i][j]=m[i-1][j];
				else
					m[i][j]=(m[i-1][j]>m[i][j-1]?m[i-1][j]:m[i][j-1]);
			}
		}
	}
	for(int k=n-1;k>=0;k--)
	{
		if(k==0&&m[0][0]!=0)
			flag[k]=1;
		else if(m[k][k]>m[k-1][k-1])
			flag[k]=1;
		else
			flag[k]=0;
	}
	cout<<"len="<<m[n-1][n-1]<<endl;
	cout<<"the number are:";
	for(int h=0;h<n;h++)
	{
		if(flag[h])
		{
			cout<<a[h]<<" ";
		}
	}
	cout<<endl;
		
}


第二种思路

/*设计一个O(n2)时间的算法,找出由n个数组成的序列的最长单调递增子序列。*/
#include <stdio.h>
#define N 100

void prin(int i);
void fun(int n);
int p[N],a[N];

void main()
{
	int i,n;
	printf("input n:");
	scanf("%d",&n);
	printf("input the array.\n");
	for(i=0;i<n;i++)
		scanf("%d",&a[i]);
	fun(n);

}

void fun(int n)
{
	int m[N];
	int i,k,max;
	m[0]=1;
	p[0]=-1;
	for(i=1;i<n;i++)
	{
		max=0;
		p[i]=-1;
		for(k=0;k<i;k++)
		{
			if(m[k]>max&&a[k]<a[i])
			{
				p[i]=k;
				max=m[k];                   
			}
		}
		m[i]=max+1;
	}
	i=0;
	for(k=1;k<n;k++)
	{
		if(m[k]>m[i])
		{
			i=k;
		}
	}	
	prin(i);
	printf("\nlen=%d\n",m[i]);

}

void prin(int i)
{
	if(p[i]<0)
	{
		printf("%d",a[i]);
		return;
	}
	prin(p[i]);
	printf(",%d",a[i]);
}



  • 2
    点赞
  • 23
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 可以使用动态规划来解决这个问题。 定义一个数组dp,其中dp[i]表示以第i个元素为结尾的最长单调递增子序列的长度。 初始化dp数组为1,因为每个元素本身都可以作为一个长度为1的单调递增子序列。 然后从第二个元素开始遍历整个序列,对于每个元素i,再从第一个元素到第i-1个元素中找到比它小的元素j,如果存在这样的元素j,则更新dp[i]为dp[j]+1,表示以j为结尾的最长单调递增子序列再加上i这个元素可以构成以i为结尾的更长的单调递增子序列。 最后,遍历整个dp数组,找到其中最大的值,即为整个序列最长单调递增子序列的长度。 时间复杂度为O(n^2)。 ### 回答2: 最长单调递增子序列(Longest Increasing Subsequence,LIS)是一道经典的动态规划问题,可以使用O(n2)或O(nlogn)的方法求解。 以下是一个O(n2)的DP算法: 设dp[i]表示以第i个元素结尾的最长单调递增子序列的长度,初始化dp[i]=1(因为以第i个元素结尾的最短递增序列长度为1)。 对于每个j<i,如果nums[j]<nums[i],说明第j个元素可以接在第i个元素后面构成递增序列,此时更新dp[i]=max(dp[i],dp[j]+1)。 最后遍历dp数组,找到最大值即可。 具体实现如下: int lengthOfLIS(vector<int>& nums) { int n=nums.size(),res=1; vector<int> dp(n,1); for(int i=1;i<n;i++){ for(int j=0;j<i;j++){ if(nums[j]<nums[i]) dp[i]=max(dp[i],dp[j]+1); } res=max(res,dp[i]); } return res; } 该算法时间复杂度为O(n2),空间复杂度为O(n)。 ### 回答3: 一、问题分析 题目要求我们设计一个o(n2)时间算法找出由n个数组成序列最长单调递增子序列。 首先我们需要明确,什么是子序列?什么是最长单调递增子序列? 1. 子序列:在给定序列中,选择任意数量的数字进行排列组合,而这些数字按照原始序列中的顺序,组成的新序列称为原序列子序列。 2. 最长单调递增子序列:在原序列中找到一组连续的元素,使得它们按原序列的顺序从小到大排列,并且满足长度最长。 我们需要设计算法,即为找出给定序列最长单调递增子序列。 二、算法设计 1. 定义状态 我们定义状态数组dp[i]为以第i个元素作为结尾的最长递增子序列长度。 2. 初始化 对状态数组进行初始化,将dp[i]置为1,因为我们可以将每个元素看做是一个长度为1的递增子序列。 3. 状态转移方程 对于下标i和j,若i<j且a[i]<a[j],则dp[j] = max(dp[j], dp[i]+1)。 这个方程的含义是,如果第i个元素比第j个元素小,并且以第i个元素为结尾的最长递增子序列加上第j个元素的话,能够得到以第j个元素结尾的递增子序列的长度,则将dp[j]更新为dp[i]+1。 在这个过程中,我们会从前往后逐个计算状态,并且在每次计算dp[j]时,都需要通过遍历0~j-1来计算其值,所以时间复杂度为o(n2)。 4. 计算最长单调递增子序列长度 对状态数组进行遍历,找出其中的最大值即为给定序列最长单调递增子序列长度。同时,我们也可以通过倒推出具体的递增子序列内容。 三、算法实现 下面是算法的具体实现: ```python def longestIncreasingSubsequence(nums): n = len(nums) dp = [1] * n for j in range(n): for i in range(j): if nums[i] < nums[j]: dp[j] = max(dp[j], dp[i] + 1) return max(dp), getSubsequence(nums, dp) def getSubsequence(nums, dp): n = len(nums) length = max(dp) index = dp.index(length) subsequence = [nums[index]] while length > 1: for i in range(index-1, -1, -1): if nums[i] < nums[index] and dp[i] == length-1: subsequence.append(nums[i]) index = i length -= 1 break return subsequence[::-1] ``` 四、总结 最长单调递增子序列问题可以通过动态规划的方式来解决,时间复杂度为o(n2)。在实际应用中,我们可以使用这个算法来解决一些子问题,比如求解二维点集中的最大递增子集合。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值