博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Codeforces Round #384 (Div. 2) E. Vladik and cards 状压dp
阅读量:5323 次
发布时间:2019-06-14

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

E. Vladik and cards

题目链接

题面

Vladik was bored on his way home and decided to play the following game. He took n cards and put them in a row in front of himself. Every card has a positive integer number not exceeding 8 written on it. He decided to find the longest subsequence of cards which satisfies the following conditions:

the number of occurrences of each number from 1 to 8 in the subsequence doesn't differ by more then 1 from the number of occurrences of any other number. Formally, if there are ck cards with number k on them in the subsequence, than for all pairs of integers the condition |ci - cj| ≤ 1 must hold.

if there is at least one card with number x on it in the subsequence, then all cards with number x in this subsequence must form a continuous segment in it (but not necessarily a continuous segment in the original sequence). For example, the subsequence [1, 1, 2, 2] satisfies this condition while the subsequence [1, 2, 2, 1] doesn't. Note that [1, 1, 2, 2] doesn't satisfy the first condition.
Please help Vladik to find the length of the longest subsequence that satisfies both conditions.

输入

The first line contains single integer n (1 ≤ n ≤ 1000) — the number of cards in Vladik's sequence.

The second line contains the sequence of n positive integers not exceeding 8 — the description of Vladik's sequence.

输出

Print single integer — the length of the longest subsequence of Vladik's sequence that satisfies both conditions.

样例输入

3

1 1 1

样例输出

1

题意

给你n个数字,你需要找到一个最长的子序列,满足以下要求:

1.对于每个i和j,要求abs(num[i]-num[j])<=1,num[i]表示这个数字i出现的次数

2.所有相同的数字应该挨在一起。

求最长的子序列长度

题解

枚举每个数字的长度num,那么显然每个数字要么是num,要么就是num+1

然后我们对于每个长度进行check就好了

dp[i][j]表示当前状态为i的时候,其中有i个数的长度为num+1,用一个next进行转移就好了

next[i][j][k]表示从i开始,j出现k次的位置是啥位置。

代码

#include
using namespace std;const int maxn = 1050;int dp[maxn][9];int nxt[maxn][9][maxn],a[maxn],n,cnt[9];int main(){ scanf("%d",&n); for(int i=1;i<=n;i++) scanf("%d",&a[i]); for(int i=1;i<=n;i++){ for(int j=1;j<=8;j++)cnt[j]=0; for(int j=1;j<=8;j++)for(int k=1;k<=n;k++) nxt[i][j][k]=1e9; for(int j=i;j<=n;j++){ cnt[a[j]]++; nxt[i][a[j]][cnt[a[j]]]=j; } } int ans=0; for(int num=0;num*8<=n;num++){ for(int i=0;i<256;i++) for(int k=0;k<=8;k++) dp[i][k]=1e9; dp[0][0]=1; for(int i=0;i<256;i++){ for(int j=0;j<=8;j++){ for(int k=1;k<=8;k++){ if((1<<(k-1))&i)continue; if(dp[i][j]>n)continue; dp[i^(1<<(k-1))][j]=min(dp[i^(1<<(k-1))][j],nxt[dp[i][j]][k][num]+1); dp[i^(1<<(k-1))][j+1]=min(dp[i^(1<<(k-1))][j+1],nxt[dp[i][j]][k][num+1]+1); } } } for(int i=0;i<=8;i++)if(dp[255][i]<=n+1)ans=max(ans,8*num+i); } cout<
<

转载于:https://www.cnblogs.com/qscqesze/p/6193189.html

你可能感兴趣的文章
CDH版本大数据集群下搭建的Hue详细启动步骤(图文详解)
查看>>
巧用Win+R
查看>>
浅析原生js模仿addclass和removeclass
查看>>
Python中的greenlet包实现并发编程的入门教程
查看>>
java中遍历属性字段及值(常见方法)
查看>>
深入理解jQuery框架-框架结构
查看>>
YUI3自动加载树实现
查看>>
python知识思维导图
查看>>
当心JavaScript奇葩的逗号表达式
查看>>
App Store最新审核指南(2015年3月更新版)
查看>>
织梦MIP文章内容页图片适配百度MIP规范
查看>>
点击复制插件clipboard.js
查看>>
[Kali_BT]通过低版本SerialPort蓝牙渗透功能手机
查看>>
C语言学习总结(三) 复杂类型
查看>>
HNOI2018
查看>>
【理财】关于理财的网站
查看>>
Ubunt中文乱码
查看>>
《当幸福来敲门》读后
查看>>
【转】系统无法进入睡眠模式解决办法
查看>>
省市县,循环组装,整合大数组
查看>>