博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
【bzoj4300】绝世好题 dp
阅读量:4571 次
发布时间:2019-06-08

本文共 999 字,大约阅读时间需要 3 分钟。

题目描述

给定一个长度为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 
#include
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;}

 

转载于:https://www.cnblogs.com/GXZlegend/p/6498432.html

你可能感兴趣的文章
数据库基础查询
查看>>
Eclipse安装SVN
查看>>
Linux性能优化建议
查看>>
OTL翻译(3) -- OTL的主要类
查看>>
java当中的定时器的4种使用方式
查看>>
hive
查看>>
Java集合排序(面试必考点之一)
查看>>
Tsql 获取服务器信息
查看>>
给laravel项目集成支付宝
查看>>
安装prometheus+grafana监控mysql redis kubernetes等
查看>>
Java眼中的XML--文件读取--1 应用DOM方式解析XML
查看>>
C++使用递归函数计算阶乘
查看>>
linker command failed with exit code 1 (use -v to see invocation)报错解决办法
查看>>
通用EF框架
查看>>
5-Java多态性理解
查看>>
vray学习笔记(4)混合材质是个什么东西
查看>>
Redis Linux版安装详解
查看>>
软件工程个人作业02------即01的升级版本
查看>>
博弈论 斯坦福game theory stanford week 3.1_
查看>>
HDU1286+线性筛素数
查看>>