【bzoj4300】绝世好题 dp

题目描述

给定一个长度为n的数列ai,求ai的子序列bi的最长长度,满足bi&bi-1!=0(2<=i<=len)。

输入

输入文件共2行。
第一行包括一个整数n。
第二行包括n个整数,第i个整数表示ai。

输出

输出文件共一行。
包括一个整数,表示子序列bi的最长长度。

样例输入

3
1 2 3

样例输出

2


题解

dp

设f[i]为选i时前i个元素的最多个数。

那么就有f[i]=max{f[j]}+1 (a[j]&a[i]!=0)

这样会TLE,于是想优化。

如果a&b!=0,根据定义,a、b的二进制数中至少有一位都为1。

那么我们可以开一个辅助数组maxn[k],记录一下所有a[i]中二进制第k位为1的f[i]的最大值。

然后扫一遍每个a[i]的数位,取最大值加到f[i]里并更新即可。

#include <cstdio>
#include <algorithm>
using namespace std;
int a[100010] , f[100010] , maxn[32];
int getnum(int n)
{
	int i;
	for(i = 0 ; i < 31 ; i ++ )
		if((1 << i) == n)
			return i;
	return 0;
}
int main()
{
	int n , i , j , t , ans = 0;
	scanf("%d" , &n);
	for(i = 1 ; i <= n ; i ++ )
		scanf("%d" , &a[i]);
	for(i = 1 ; i <= n ; i ++ )
	{
		f[i] = 1;
		for(j = a[i] ; j ; j -= j & (-j))
			f[i] = max(f[i] , maxn[getnum(j & (-j))] + 1);
		for(j = a[i] ; j ; j -= j & (-j))
		{
			t = getnum(j & (-j));
			maxn[t] = max(maxn[t] , f[i]);
		}
		ans = max(ans , f[i]);
	}
	printf("%d\n" , ans);
	return 0;
}

 

posted @ 2017-03-03 21:04  GXZlegend  阅读(352)  评论(0编辑  收藏  举报