POJ 1952 BUY LOW, BUY LOWER DP记录数据

最长递减子序列。加记录有多少个最长递减子序列。然后须要去重。

最麻烦的就是去重了。

主要的思路就是:全面出现反复的值,然后还是同样长度的子序列。这里的DP记录的子序列是以当前值为结尾的时候,而且一定选择这个值的最长递减子序列。 那么就须要减去前面已经出现过了的子序列。

有点绕口。

举例就是9 8 9 8 2 和 10 5 12 5 3;这些样例去重。

本类型的题目假设不用记录数据是能够使用O(nlgn)的算法的,只是临时不知道怎样记录数据。故此这里仅仅使用DP了。


#include <stdio.h>
#include <vector>
#include <string.h>
#include <algorithm>
#include <iostream>
#include <string>
#include <limits.h>
#include <stack>
#include <queue>
#include <set>
#include <map>
using namespace std;
const int MAX_N = 5001;
int arr[MAX_N], N, tbl[MAX_N], C[MAX_N];

void getLongest(int &len, int &n)
{
	memset(tbl, 0, sizeof(int) * (N+1));
	memset(C, 0, sizeof(int) * (N+1));
	tbl[0] = 1; C[0] = 1;
	for (int i = 1; i < N; i++)
	{
		tbl[i] = 1;
		for (int j = 0; j < i; j++)
		{
			if (tbl[j] == -1) continue;
			if (arr[j] > arr[i] && tbl[i] < tbl[j]+1)
			{
				tbl[i] = tbl[j]+1;
			}
		}
		for (int j = 0; j < i; j++)
		{
			if (arr[j] > arr[i] && tbl[i] == tbl[j]+1)
			{
				C[i] += C[j];
			}
		}
		if (C[i] == 0) C[i] = 1;//递增的时候
		/*能够不用以下这段代码
		for (int j = 0; j < i; j++)
		{
			if (arr[i] == arr[j] && tbl[i] == tbl[j] && C[i] == C[j])
			{
				tbl[i] = -1;
				break;
			}//去掉同样的数据 9 8 9 8
		}
		if (tbl[i] == -1) continue;*/

		for (int j = 0; j < i; j++)
		{
			if (arr[j] == arr[i] && tbl[j] == tbl[i]) C[i] -= C[j];
		}//特例:6 5 7 5 3 须要去掉前后5反复的地方
	}
	len = INT_MIN;
	for (int i = 0; i < N; i++)
	{
		len = max(len, tbl[i]);
	}
	n = 0;
	for (int i = 0; i < N; i++)
	{
		if (tbl[i] == len) n += C[i];
	}
}

int main()
{
	while (scanf("%d", &N) != EOF)
	{
		for (int i = 0; i < N; i++)
		{
			scanf("%d", arr + i);
		}
		int len, n;
		getLongest(len, n);
		printf("%d %d\n", len, n);
	}
	return 0;
}



转载于:https://www.cnblogs.com/zhchoutai/p/7128282.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值