[Swust OJ 581]--彩色的石子(状压dp)

题目链接:http://acm.swust.edu.cn/problem/0581/

Time limit(ms): 1000        Memory limit(kb): 65535

Description

把m个含有k种不同颜色的石子放成一条线上。现在要问你怎么才能取走 
最少的石子,使得没有两个相同颜色的石子之间含有其它的颜色

Input

有多组测试数据,每组测试数据有两行,第一行是m和k,1<=m<=100, 
1<=k<=5,第二行就是那m个石子的颜色,用1到k表示,最后输入0 0表示 
程序结束。

Output

对于每组测试数据,请你找出最少取走的石子数目并输出

Sample Input


10 3

2 1 2 2 1 1 3 1 3 3

0 0

Sample Output

2

解题思路:状态压缩dp,一共最多5类石子,那么

(1)dp[1<<5][5]用来表示(二进制每一位对应)是否含有第k类石子,表示以第k类石子结尾含有1<<5(二进制数)每一位对应种类石子下的可行序列最大值

(2)题目中给出的种类是从1开始的这里我们要转化为从0开始否则1<<6浪费空间(算一个小小的优化吧~~~)

(3)在0-1<<5中查找不同的状态i&(1 << x) 表明前面含有x类石,那么可以直接dp[i][x]++

(4)在(3)执行完以后,如果!(i&(1 << x)) 加一生成的新段如果此时dp值大于相同状态的值,更新即

for (L = 0; L < k; L++){
  if (L != x && dp[j | (1 << x)][x] < dp[j][L] + 1) //不含x类的段加一生成的新段如果大于相同状态的值,更新
      dp[j | (1 << x)][x] = dp[j][L] + 1;
   }

(5)最后查询所有dp[1<<k-1][i]状态下的最长可行序列,用总长度减去即可

dp方程总结如下

每输入一个x

if(s&(1<<x)
    dp[s][x] = dp[s][x] + 1
else
    dp[1<<x|s][x] = max(dp[1<<x|s][x],dp[s][L]+1)

L(0,k)

  

代码如下:

 1 #include <iostream>
 2 #include <cstring>
 3 using namespace std;
 4 int main(){
 5     int m, k, dp[1 << 5][5];
 6     //dp[][k] 二进制每一位对应含有第k类石子,表示以第k类石子结尾含有1<<5每一位对应下石子的最大值
 7     while (cin >> m >> k, m || k){
 8         int i, x, j, L, t = 1 << k, maxn = 0;
 9         memset(dp, 0, sizeof(dp));
10         for (i = 0; i < m; i++){
11             cin >> x;
12             x--;
13             for (j = 0; j < t; j++){
14                 if (j&(1 << x)) //对于前面也是以x类石子结尾的最大值直接加1
15                     dp[j][x]++;
16             }
17             for (j = 0; j < t; j++){
18                 if (!(j&(1 << x)))  //前面不含x类石子
19                 for (L = 0; L < k; L++){
20                     if (L != x && dp[j | (1 << x)][x] < dp[j][L] + 1) //不含x类的段加一生成的新段如果大于相同状态的值,更新
21                         dp[j | (1 << x)][x] = dp[j][L] + 1;
22                 }
23             }
24         }
25         for (i = 0; i<k; i++)
26         if (dp[(1 << k) - 1][i]>maxn)
27             maxn = dp[(1 << k) - 1][i];
28         cout << m - maxn << endl;
29     }
30     return 0;
31 }

时间: 2024-10-22 08:20:10

[Swust OJ 581]--彩色的石子(状压dp)的相关文章

西电oj 1038 状压dp

西电oj 1038  状压dp 1038: 裁玻璃 时间限制: 1 Sec  内存限制: 128 MB提交: 33  解决: 4[提交][状态][讨论版] 题目描述 张老板的玻璃店开张了,生意火爆.今天,隔壁玻璃店的刘老板拿来一块玻璃,意在刁难张老板.刘老板说:“我这块玻璃是由N(行)*M(列)小正方形玻璃拼成的,但是其中有一些玻璃坏了,我希望你现在把它裁成尽量多的2*2的小玻璃,而且这些小玻璃都不能有坏的地方.如果你裁出来的块数不是最多的,我就把你赶出建材市场.”现在,张老板来拜托你帮他解决这

(九度OJ)题目1338:角斗士(状压DP)

题目描述: 角斗士是古罗马奴隶社会的一种特殊身份的奴隶,他们的职责是在角斗场上进行殊死搏斗,为了人们提供野蛮的娱乐.他们的结局或是战死,或者由于表现突出赢得胜利而获得释放. 现在在角斗场里有N个待战的角斗士(1 <=N<=18),每天都将举行一场比赛,为了保持比赛的刺激性,每场比赛前才会在所有当前活着的角斗士之中随机选择两名进行上场拼杀.每场比赛的结束条件即为其中一名被杀死.当进行了N场比赛之后,最后存活的角斗士将被释放.而你将被赋予一个任务,计算出每名角斗士最终存活的概率.我们将提供角斗士之

HDU 5067 Harry And Dig Machine(状压DP)(TSP问题)

题目地址:pid=5067">HDU 5067 经典的TSP旅行商问题模型. 状压DP. 先分别预处理出来每两个石子堆的距离.然后将题目转化成10个城市每一个城市至少经过一次的最短时间模型.然后简单的状压DP就可以. 代码例如以下: #include <iostream> #include <cstdio> #include <string> #include <cstring> #include <stdlib.h> #incl

状压dp,区间dp,矩阵快速幂

DP 首先先回忆一下dp,dp叫做记忆化搜索,是一种可以把暴力搜索中重复的部分重复利用,从而到达减小复杂度的目的.比如最应该熟悉的背包模型,如果你把选择的过程看成一步一步的,那么在这么多的搜索路径中一定有着很多很多的重复部分,dp就是一种把重复的部分加以利用的方法.相信大家都已经在以前的练习中已经明白了dp是什么样的思路了,接下来的两种dp会在大家已经了解经典的背包dp等模型下展开. 状态压缩dp: 首先先讲一个状压dp最最经典的模型,求哈密尔顿路径问题,也叫做旅行商问题:给你一张图,你要在每个

ZOJ3305Get Sauce 状压DP,

状压DP的题目留个纪念,首先题意一开始读错了,搞了好久,然后弄好了,觉得DFS可以,最后超时,修改了很久还是超时,没办法看了一下n的范围,然后觉得状压可以,但是没有直接推出来,就记忆化搜索了一下,可是一直错,莫名奇妙,然后没办法看了一下题解,发现了下面这个比较好的方法,然后按照这个方程去推,然后敲,也是WA了好多把,写的太搓了,没人家的清楚明了,唉~也算是给自己留个纪念,状压一直做的都不太好~唉~还好理解了, 参考了  http://blog.csdn.net/nash142857/articl

poj 2411 Mondriaan&#39;s Dream(状压DP)

Mondriaan's Dream Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 12232   Accepted: 7142 Description Squares and rectangles fascinated the famous Dutch painter Piet Mondriaan. One night, after producing the drawings in his 'toilet series

(状压dp)uva 10817 Headmaster&#39;s Headache

题目地址 1 #include <bits/stdc++.h> 2 typedef long long ll; 3 using namespace std; 4 const int MAX=1e5+5; 5 const int INF=1e9; 6 int s,m,n; 7 int cost[125]; 8 //char sta[MAX]; 9 string sta; 10 int able[125]; 11 int dp[125][1<<8][1<<8]; 12 in

HDU5816 Hearthstone(状压DP)

题目 Source http://acm.hdu.edu.cn/showproblem.php?pid=5816 Description Hearthstone is an online collectible card game from Blizzard Entertainment. Strategies and luck are the most important factors in this game. When you suffer a desperate situation an

HDU 4336 容斥原理 || 状压DP

状压DP :F(S)=Sum*F(S)+p(x1)*F(S^(1<<x1))+p(x2)*F(S^(1<<x2))...+1; F(S)表示取状态为S的牌的期望次数,Sum表示什么都不取得概率,p(x1)表示的是取x1的概率,最后要加一因为有又多拿了一次.整理一下就可以了. 1 #include <cstdio> 2 const int Maxn=23; 3 double F[1<<Maxn],p[Maxn]; 4 int n; 5 int main() 6