状态压缩DP总结(谨记大牛总结)

2024-05-11 03:32

本文主要是介绍状态压缩DP总结(谨记大牛总结),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

状态压缩DP总结【POJ3254】【POJ1185】【POJ3311】【HDU3001】【POJ2288】【ZOJ4257】【POJ2411】【HDU3681】

分类: 动态规划 10311人阅读 评论(11) 收藏 举报
vb path forms c struct each

动态规划本来就很抽象,状态的设定和状态的转移都不好把握,而状态压缩的动态规划解决的就是那种状态很多,不容易用一般的方法表示的动态规划问题,这个就更加的难于把握了。难点在于以下几个方面:状态怎么压缩?压缩后怎么表示?怎么转移?是否具有最优子结构?是否满足后效性?涉及到一些位运算的操作,虽然比较抽象,但本质还是动态规划。找准动态规划几个方面的问题,深刻理解动态规划的原理,开动脑筋思考问题。这才是掌握动态规划的关键。

动态规划最关键的要处理的问题就是位运算的操作,容易出错,状态的设计也直接决定了程序的效率,或者代码长短。状态转移方程一定要仔细推敲,不可一带而过,要思考为什么这么做,掌握一个套路,遇见这类问题能快速的识别出问题的本质,找出状态转移方程和DP的边界条件。

下面是一些关于状态压缩DP的题目,大都不难。状压DP的东西还有很多,还会接着总结。

【POJ3254】Corn Fields

【题目大意】一个矩阵里有很多格子,每个格子有两种状态,可以放牧和不可以放牧,可以放牧用1表示,否则用0表示,在这块牧场放牛,要求两个相邻的方格不能同时放牛,即牛与牛不能相邻。问有多少种放牛方案(一头牛都不放也是一种方案)

【解析】根据题意,把每一行的状态用二进制的数表示,0代表不在这块放牛,1表示在这一块放牛。首先很容易看到,每一行的状态要符合牧场的硬件条件,即牛必须放在能放牧的方格上。这样就能排除一些状态。另外,牛与牛之间不能相邻,这样就要求每一行中不能存在两个相邻的1,这样也能排除很多状态。然后就是根据上一行的状态转移到当前行的状态的问题了。必须符合不能有两个1在同一列(两只牛也不能竖着相邻)的条件。这样也能去掉一些状态。然后,上一行的所有符合条件的状态的总的方案数就是当前行该状态的方案数。

【状态表示】dp[state][i]:在状态为state时,到第i行符合条件的可以放牛的方案数

【状态转移方程】dp[state][i] =Sigma dp[state'][i-1] (state'为符合条件的所有状态)

【DP边界条件】首行放牛的方案数dp[state][1] =1(state符合条件) OR 0 (state不符合条件)

【代码】

[cpp] view plain copy print ?
  1. #include <cstdio>   
  2. #include <cstring>   
  3. using namespace std;  
  4. #define mod 100000000   
  5. int M,N,top = 0;  
  6. int state[600],num[110];  
  7. int dp[20][600];  
  8. int cur[20];  
  9. inline bool ok(int x){  
  10.    if(x&x<<1)return 0;  
  11.    return 1;  
  12. }  
  13. void init(){  
  14.    top = 0;  
  15.    int total = 1 << N;  
  16.    for(int i = 0; i < total; ++i){  
  17.        if(ok(i))state[++top] = i;  
  18.    }  
  19. }  
  20. inline bool fit(int x,int k){  
  21.    if(x&cur[k])return 0;  
  22.    return 1;  
  23. }  
  24. inline int jcount(int x)  
  25. {  
  26.    int cnt=0;  
  27.    while(x)  
  28.    {  
  29.        cnt++;  
  30.        x&=(x-1);  
  31.    }  
  32.    return cnt;  
  33. }  
  34.    
  35. int main(){  
  36.     while(scanf("%d%d",&M,&N)!= EOF){  
  37.        init();  
  38.        memset(dp,0,sizeof(dp));  
  39.        for(int i = 1; i <= M; ++i){  
  40.            cur[i] = 0;  
  41.            int num;  
  42.            for(int j = 1; j <= N; ++j){  
  43.                 scanf("%d",&num);  
  44.                if(num == 0)cur[i] +=(1<<(N-j));  
  45.            }  
  46.        }  
  47.        for(int i = 1;i <= top;i++){  
  48.            if(fit(state[i],1)){  
  49.                 dp[1][i] = 1;  
  50.            }  
  51.        }  
  52.        for(int i = 2; i <= M; ++i){  
  53.            for(int k = 1; k <= top; ++k){  
  54.                 if(!fit(state[k],i))continue;  
  55.                 for(int j = 1; j <= top ;++j){  
  56.                    if(!fit(state[j],i-1))continue;  
  57.                    if(state[k]&state[j])continue;  
  58.                     dp[i][k] = (dp[i][k] +dp[i-1][j])%mod;  
  59.                 }  
  60.            }  
  61.        }  
  62.        int ans = 0;  
  63.        for(int i = 1; i <= top; ++i){  
  64.            ans = (ans + dp[M][i])%mod;  
  65.        }  
  66.        printf("%d\n",ans);  
  67.    }  
  68. }  
#include <cstdio>
#include <cstring>
using namespace std;
#define mod 100000000
int M,N,top = 0;
int state[600],num[110];
int dp[20][600];
int cur[20];
inline bool ok(int x){
if(x&x<<1)return 0;
return 1;
}
void init(){
top = 0;
int total = 1 << N;
for(int i = 0; i < total; ++i){
if(ok(i))state[++top] = i;
}
}
inline bool fit(int x,int k){
if(x&cur[k])return 0;
return 1;
}
inline int jcount(int x)
{
int cnt=0;
while(x)
{
cnt++;
x&=(x-1);
}
return cnt;
}
int main(){
while(scanf("%d%d",&M,&N)!= EOF){
init();
memset(dp,0,sizeof(dp));
for(int i = 1; i <= M; ++i){
cur[i] = 0;
int num;
for(int j = 1; j <= N; ++j){
scanf("%d",&num);
if(num == 0)cur[i] +=(1<<(N-j));
}
}
for(int i = 1;i <= top;i++){
if(fit(state[i],1)){
dp[1][i] = 1;
}
}
for(int i = 2; i <= M; ++i){
for(int k = 1; k <= top; ++k){
if(!fit(state[k],i))continue;
for(int j = 1; j <= top ;++j){
if(!fit(state[j],i-1))continue;
if(state[k]&state[j])continue;
dp[i][k] = (dp[i][k] +dp[i-1][j])%mod;
}
}
}
int ans = 0;
for(int i = 1; i <= top; ++i){
ans = (ans + dp[M][i])%mod;
}
printf("%d\n",ans);
}
}


【POJ1185】炮兵阵地--经典状压DP

【题目大意】类似于上面一道题,一个方格组成的矩阵,每个方格可以放大炮用0表示,不可以放大炮用1表示(原题用字母),让放最多的大炮,大炮与大炮间不会互相攻击。

 

【解析】可以发现,对于每一行放大炮的状态,只与它上面一行和上上一行的状态有关,每一行用状态压缩的表示方法,0表示不放大炮,1表示放大炮,同样的,先要满足硬件条件,即有的地方不能放大炮,然后就是每一行中不能有两个1的距离小于2(保证横着不互相攻击),这些要预先处理一下。然后就是状态表示和转移的问题了,因为是和前两行的状态有关,所以要开个三维的数组来表示状态,当前行的状态可由前两行的状态转移而来。即如果当前行的状态符合前两行的约束条件(不和前两行的大炮互相攻击),则当前行的最大值就是上一个状态的值加上当前状态中1的个数(当前行放大炮的个数) 

【状态表示】dp[i][j][k] 表示第i行状态为k,第i-1状态为j时的最大炮兵个数。 

【状态转移方程】dp[i][k][t] =max(dp[i][k][t],dp[i-1][j][k]+num[t]); num[t]为t状态中1的个数 

【DP边界条件】dp[1][1][i] =num[i] 状态i能够满足第一行的硬件条件(注意:这里的i指的是第i个状态,不是一个二进制数,开一个数组保存二进制状态) 

【代码】

[cpp] view plain copy print ?
  1. #include <cstdio>   
  2. #include <cstring>   
  3. using namespace std;  
  4. #define max(a,b) (a) > (b) ? (a) : (b)   
  5.    
  6. int N,M;  
  7. char map[110][20],num[110],top;  
  8. int stk[70],cur[110];  
  9. int dp[110][70][70];  
  10.    
  11. inline bool ok(int x){  //判断该状态是否合法,即不存在相邻的1之间的距离小于3的   
  12.    if(x&(x<<1)) return 0;  
  13.    if(x&(x<<2)) return 0;  
  14.    return 1;  
  15. }  
  16. //找到所有可能的合法状态,最多60种   
  17. inline void jinit()  
  18. {  
  19.    top=0;  
  20.    int i,total=1<<N;  
  21.    for(i=0;i<total;i++) if(ok(i)) stk[++top]=i;  
  22. }  
  23. //判断状态x是否与第k行匹配   
  24. inline bool fit(int x,int k)  
  25. {  
  26.    if(cur[k]&x) return 0;  
  27.    return 1;  
  28. }  
  29. //数一个整型数x的二进制中1的个数(用于初始化)   
  30. inline int jcount(int x)  
  31. {  
  32.    int cnt=0;  
  33.    while(x)  
  34.    {  
  35.        cnt++;  
  36.        x&=(x-1);  
  37.    }  
  38.    return cnt;  
  39. }  
  40.    
  41. int main(){  
  42.    while(scanf("%d%d",&M,&N) != EOF){  
  43.        if(N == 0 && M == 0)break;  
  44.        jinit();  
  45.        for(int i = 1; i <= M; ++i)scanf("%s",map[i]+1);  
  46.        for(int i = 1; i <= M; ++i)  
  47.        for(int j = 1; j <= N; ++j){  
  48.            cur[i]=0;  
  49.            for(j=1;j<=N;j++){  
  50.                 if(map[i][j]=='H')cur[i]+=(1<<(j-1));  
  51.            }  
  52.        }  
  53.        memset(dp,-1,sizeof(dp));  
  54.    
  55.        //初始化第一行状态   
  56.        for(int i = 1;i <= top;i++){  
  57.            num[i]=jcount(stk[i]);  
  58.            if(fit(stk[i],1))  
  59.                 dp[1][1][i]=num[i];  
  60.        }  
  61.        int i,t,j,k;  
  62.        for(i = 2;i <= M;i++){  
  63.            for(t = 1;t <= top;t++){  
  64.                 if(!fit(stk[t],i)) continue;  
  65.                 for(j = 1;j <= top;j++)  
  66.                 {  
  67.                     if(stk[t]&stk[j])continue;  
  68.                     for(k = 1;k <= top;k++)  
  69.                     {  
  70.                         if(stk[t]&stk[k])continue;  
  71.                         if(dp[i-1][j][k]==-1)continue;  
  72.                         dp[i][k][t] =max(dp[i][k][t],dp[i-1][j][k]+num[t]);  
  73.                     }  
  74.                 }  
  75.            }  
  76.        }  
  77.        int ans = 0;  
  78.        for(i = 1; i <= M; ++i)  
  79.        for(j = 1; j <= top; ++j)  
  80.        for(k = 1; k <= top; ++k)  
  81.        ans = max(ans,dp[i][j][k]);  
  82.        printf("%d\n",ans);  
  83.    }  
  84.    return 0;  
  85. }  
#include <cstdio>
#include <cstring>
using namespace std;
#define max(a,b) (a) > (b) ? (a) : (b)
int N,M;
char map[110][20],num[110],top;
int stk[70],cur[110];
int dp[110][70][70];
inline bool ok(int x){  //判断该状态是否合法,即不存在相邻的1之间的距离小于3的
if(x&(x<<1)) return 0;
if(x&(x<<2)) return 0;
return 1;
}
//找到所有可能的合法状态,最多60种
inline void jinit()
{
top=0;
int i,total=1<<N;
for(i=0;i<total;i++) if(ok(i)) stk[++top]=i;
}
//判断状态x是否与第k行匹配
inline bool fit(int x,int k)
{
if(cur[k]&x) return 0;
return 1;
}
//数一个整型数x的二进制中1的个数(用于初始化)
inline int jcount(int x)
{
int cnt=0;
while(x)
{
cnt++;
x&=(x-1);
}
return cnt;
}
int main(){
while(scanf("%d%d",&M,&N) != EOF){
if(N == 0 && M == 0)break;
jinit();
for(int i = 1; i <= M; ++i)scanf("%s",map[i]+1);
for(int i = 1; i <= M; ++i)
for(int j = 1; j <= N; ++j){
cur[i]=0;
for(j=1;j<=N;j++){
if(map[i][j]=='H')cur[i]+=(1<<(j-1));
}
}
memset(dp,-1,sizeof(dp));
//初始化第一行状态
for(int i = 1;i <= top;i++){
num[i]=jcount(stk[i]);
if(fit(stk[i],1))
dp[1][1][i]=num[i];
}
int i,t,j,k;
for(i = 2;i <= M;i++){
for(t = 1;t <= top;t++){
if(!fit(stk[t],i)) continue;
for(j = 1;j <= top;j++)
{
if(stk[t]&stk[j])continue;
for(k = 1;k <= top;k++)
{
if(stk[t]&stk[k])continue;
if(dp[i-1][j][k]==-1)continue;
dp[i][k][t] =max(dp[i][k][t],dp[i-1][j][k]+num[t]);
}
}
}
}
int ans = 0;
for(i = 1; i <= M; ++i)
for(j = 1; j <= top; ++j)
for(k = 1; k <= top; ++k)
ans = max(ans,dp[i][j][k]);
printf("%d\n",ans);
}
return 0;
}


【POJ3311】Hie With The Pie

【题目大意】类似于TSP问题,只是每个点可以走多次,比经典TSP问题不同的是要先用弗洛伊的预处理一下两两之间的距离。求最短距离。

【解析】可以用全排列做,求出一个最短的距离即可。或者用状态压缩DP.用一个二进制数表示城市是否走过

【状态表示】dp[state][i]表示到达i点状态为state的最短距离

【状态转移方程】dp[state][i] =min{dp[state][i],dp[state'][j]+dis[j][i]} dis[j][i]为j到i的最短距离

【DP边界条件】dp[state][i] =dis[0][i]  state是只经过i的状态

【代码】

[cpp] view plain copy print ?
  1. #include<iostream>    
  2. #define INF 100000000    
  3. using namespace std;   
  4. int dis[12][12];   
  5. int dp[1<<11][12];   
  6. int n,ans,_min;   
  7. int main()   
  8. {   
  9.    //freopen("in.txt","r",stdin);    
  10.    while(scanf("%d",&n) && n)   
  11.    {   
  12.        for(int i = 0;i <= n;++i)   
  13.            for(int j = 0;j <= n;++j)   
  14.                scanf("%d",&dis[i][j]);   
  15.        for(int k = 0;k <= n;++k)   
  16.            for(int i = 0;i <= n;++i)   
  17.                 for(int j = 0;j <=n;++j)   
  18.                     if(dis[i][k] + dis[k][j]< dis[i][j])   
  19.                         dis[i][j] = dis[i][k] +dis[k][j];   
  20.            
  21.        for(int S = 0;S <= (1<<n)-1;++S)//枚举所有状态,用位运算表示    
  22.            for(int i = 1;i <= n;++i)   
  23.            {   
  24.                 if(S & (1<<(i-1)))//状态S中已经过城市i    
  25.                 {   
  26.                     if(S ==(1<<(i-1)))   dp[S][i] =dis[0][i];//状态S只经过城市I,最优解自然是从0出发到i的dis,这也是DP的边界   
  27.                     else//如果S有经过多个城市    
  28.                     {   
  29.                         dp[S][i] = INF;   
  30.                         for(int j = 1;j <=n;++j)   
  31.                         {   
  32.                             if(S &(1<<(j-1)) && j != i)//枚举不是城市I的其他城市    
  33.                                 dp[S][i] =min(dp[S^(1<<(i-1))][j] + dis[j][i],dp[S][i]);   
  34.                             //在没经过城市I的状态中,寻找合适的中间点J使得距离更短   
  35.                         }   
  36.                     }   
  37.                 }   
  38.             }   
  39.        ans = dp[(1<<n)-1][1] + dis[1][0];   
  40.        for(int i = 2;i <= n;++i)   
  41.            if(dp[(1<<n)-1][i] + dis[i][0] < ans)   
  42.                 ans = dp[(1<<n)-1][i] +dis[i][0];   
  43.        printf("%d\n",ans);   
  44.    }   
  45.    return 0;   
  46. }  
#include<iostream> 
#define INF 100000000 
using namespace std; 
int dis[12][12]; 
int dp[1<<11][12]; 
int n,ans,_min; 
int main() 
{ 
//freopen("in.txt","r",stdin); 
while(scanf("%d",&n) && n) 
{ 
for(int i = 0;i <= n;++i) 
for(int j = 0;j <= n;++j) 
scanf("%d",&dis[i][j]); 
for(int k = 0;k <= n;++k) 
for(int i = 0;i <= n;++i) 
for(int j = 0;j <=n;++j) 
if(dis[i][k] + dis[k][j]< dis[i][j]) 
dis[i][j] = dis[i][k] +dis[k][j]; 
for(int S = 0;S <= (1<<n)-1;++S)//枚举所有状态,用位运算表示 
for(int i = 1;i <= n;++i) 
{ 
if(S & (1<<(i-1)))//状态S中已经过城市i 
{ 
if(S ==(1<<(i-1)))   dp[S][i] =dis[0][i];//状态S只经过城市I,最优解自然是从0出发到i的dis,这也是DP的边界
else//如果S有经过多个城市 
{ 
dp[S][i] = INF; 
for(int j = 1;j <=n;++j) 
{ 
if(S &(1<<(j-1)) && j != i)//枚举不是城市I的其他城市 
dp[S][i] =min(dp[S^(1<<(i-1))][j] + dis[j][i],dp[S][i]); 
//在没经过城市I的状态中,寻找合适的中间点J使得距离更短
} 
} 
} 
} 
ans = dp[(1<<n)-1][1] + dis[1][0]; 
for(int i = 2;i <= n;++i) 
if(dp[(1<<n)-1][i] + dis[i][0] < ans) 
ans = dp[(1<<n)-1][i] +dis[i][0]; 
printf("%d\n",ans); 
} 
return 0; 
}


【HDU3001】Traveling

【题目大意】10个点的TSP问题,但是要求每个点最多走两边,不是只可以走一次,所以要用三进制的状态压缩解决这个问题。可以预处理每个状态的第k位是什么。

【解析】和tsp问题相同,类似于上面那个题

【状态表示】【状态转移方程】同上题,具体见代码

【代码】

[cpp] view plain copy print ?
  1. #include <cstdio>   
  2. #include <cstring>   
  3. #define INF 0x1f1f1f1f    //刚发现这里写0x1f1f1f跑的比0x1f1f1f1f差不多慢了一倍!Orz~   
  4. #define min(a,b) (a) < (b) ? (a) : (b)   
  5. using namespace std;  
  6.    
  7. int N,M;  
  8. int tri[12] ={0,1,3,9,27,81,243,729,2187,6561,19683,59049};  
  9. int dig[59050][11];  //dig[state][k_dig]  状态state的第k位是多少   
  10. int edge[11][11],dp[59050][11];  
  11.    
  12. int main(){  
  13.    for(int i = 0; i < 59050; ++i){  
  14.        int t = i;  
  15.        for(int j = 1; j <= 10; ++j){  
  16.            dig[i][j] = t%3;  
  17.            t /= 3;  
  18.            if(t == 0)break;  
  19.        }  
  20.    }  
  21.    
  22.    while(scanf("%d%d",&N,&M) != EOF){  
  23.        memset(edge,INF,sizeof(edge));  
  24.    
  25.       int a,b,c;  
  26.       while(M --){  
  27.           scanf("%d%d%d",&a,&b,&c);  
  28.           if(c < edge[a][b])edge[a][b] = edge[b][a] = c;  
  29.       }  
  30.    
  31.       memset(dp,INF,sizeof(dp));  
  32.    
  33.       for(int i = 1; i <= N; ++i)dp[tri[i]][i] = 0;  
  34.       int ans = INF;  
  35.       for(int S = 0; S < tri[N+1]; ++S){  
  36.           int visit_all = 1;  
  37.           for(int i = 1; i <= N; ++i){  
  38.                if(dig[S][i] == 0)visit_all = 0;  
  39.                if(dp[S][i] == INF)continue;  
  40.    
  41.                for(int j = 1; j <= N; ++j){  
  42.                    if(i == j)continue;  
  43.                    if(edge[i][j] == INF ||dig[S][j] >= 2)continue;  
  44.                    int newS = S + tri[j];  
  45.                    dp[newS][j] =min(dp[newS][j],dp[S][i] + edge[i][j]);  
  46.                }  
  47.           }  
  48.           if(visit_all){  
  49.                for(int j = 1; j <= N; ++j)  
  50.                 ans = min(ans,dp[S][j]);  
  51.           }  
  52.    
  53.       }  
  54.       if(ans == INF){  
  55.           puts("-1");  
  56.           continue;  
  57.       }  
  58.       printf("%d\n",ans);  
  59.    }  
  60.    return 0;  
  61. }  
#include <cstdio>
#include <cstring>
#define INF 0x1f1f1f1f    //刚发现这里写0x1f1f1f跑的比0x1f1f1f1f差不多慢了一倍!Orz~
#define min(a,b) (a) < (b) ? (a) : (b)
using namespace std;
int N,M;
int tri[12] ={0,1,3,9,27,81,243,729,2187,6561,19683,59049};
int dig[59050][11];  //dig[state][k_dig]  状态state的第k位是多少
int edge[11][11],dp[59050][11];
int main(){
for(int i = 0; i < 59050; ++i){
int t = i;
for(int j = 1; j <= 10; ++j){
dig[i][j] = t%3;
t /= 3;
if(t == 0)break;
}
}
while(scanf("%d%d",&N,&M) != EOF){
memset(edge,INF,sizeof(edge));
int a,b,c;
while(M --){
scanf("%d%d%d",&a,&b,&c);
if(c < edge[a][b])edge[a][b] = edge[b][a] = c;
}
memset(dp,INF,sizeof(dp));
for(int i = 1; i <= N; ++i)dp[tri[i]][i] = 0;
int ans = INF;
for(int S = 0; S < tri[N+1]; ++S){
int visit_all = 1;
for(int i = 1; i <= N; ++i){
if(dig[S][i] == 0)visit_all = 0;
if(dp[S][i] == INF)continue;
for(int j = 1; j <= N; ++j){
if(i == j)continue;
if(edge[i][j] == INF ||dig[S][j] >= 2)continue;
int newS = S + tri[j];
dp[newS][j] =min(dp[newS][j],dp[S][i] + edge[i][j]);
}
}
if(visit_all){
for(int j = 1; j <= N; ++j)
ans = min(ans,dp[S][j]);
}
}
if(ans == INF){
puts("-1");
continue;
}
printf("%d\n",ans);
}
return 0;
}


【POJ2288】Islands and Bridge

【题目大意】求汉密尔顿的一道变形问题,中间每个点有权值,关于最后得分的描述如下

Suppose there are n islands. The value of aHamilton path C1C2...Cn is calculated as the sum of three parts. Let Vi be thevalue for the island Ci. As the first part, we sum over all the Vi values foreach island in the path. For the second part, for each edge CiCi+1 in the path,we add the product Vi*Vi+1. And for the third part, whenever three consecutiveislands CiCi+1Ci+2 in the path forms a triangle in the map, i.e. there is abridge between Ci and Ci+2, we add the product Vi*Vi+1*Vi+2. 

这题要求让得分最高

【解析】发现每个点的状态由前面两个点确定,用DP(S,A,B)表示状态为S时,当前到达A,而上一个点是B时的最大得分,这个状态由DP(S',B,C)通过从B走到A得到,S'=S-(1<<A),即S'状态就是经过B和C但不经过A的一个状态,C是不同于A和B的一个点。

【状态转移】dp[S][A][B] =max(dp[S][A][B],dp[S'][B][C]+temp) 这里的temp指的是加上的得分即Vb*Va+Va,如果构成三角关系(即A和C间有边),temp就要再加上Vb*Va*Vc.

【边界条件】DP((1<<A)+(1<<B),A,B)=Va+Vb+Va*Vb(A和B间有边)表示

【代码】

[cpp] view plain copy print ?
  1. #include <cstdio>   
  2. #include <cstring>   
  3. using namespace std;  
  4. const int MAXN = 13;  
  5. const int MAX_S = 1<<(MAXN+1);  
  6. long long dp[MAX_S][MAXN+1][MAXN+1];  
  7. long long way[MAX_S][MAXN+1][MAXN+1];  
  8. int edge[MAXN+1][MAXN+1];  
  9. long long V[MAXN+1];  
  10.    
  11. int N,M;  
  12. int main(){  
  13.    int cas;  
  14.    scanf("%d",&cas);  
  15.    while(cas --){  
  16.        memset(edge,0,sizeof(edge));  
  17.        scanf("%d%d",&N,&M);  
  18.        for(int i = 1; i <= N; ++i)  
  19.            scanf("%d",&V[i]);  
  20.        if(N == 1){  
  21.            printf("%d 1\n",V[1]);  
  22.            continue;  
  23.        }  
  24.        int a,b;  
  25.        while(M --){  
  26.            scanf("%d%d",&a,&b);  
  27.            edge[a][b] = edge[b][a] = 1;  
  28.        }  
  29.    
  30.         memset(dp,-1,sizeof(dp));  
  31.        memset(way,0,sizeof(way));  
  32.        int ii,jj;  
  33.        long long temp;  
  34.        for(int i = 1; i <= N; ++i)  
  35.        for(int j = 1; j <= N; ++j){  
  36.            if(i == j || !edge[i][j])continue;  
  37.            ii = 1<<(i-1);  
  38.            jj = 1<<(j-1);  
  39.            temp = V[i]+V[j]+V[i]*V[j];  
  40.            dp[ii+jj][i][j] = temp;  
  41.            way[ii+jj][i][j] = 1;  
  42.        }  
  43.    
  44.        for(int S = 0; S < (1<<N); ++S)  
  45.        for(int i = 1; i <= N; ++i){  
  46.            if((S&(1<<(i-1))) == 0)continue;  
  47.            for(int j = 1; j <= N; ++j){  
  48.                 if((S&(1<<(j-1))) ==0 || i == j || !edge[i][j])continue;  
  49.                 for(int k = 1; k <= N; ++k){  
  50.                     if(i == k || j == k ||(S&(1<<(k-1))) == 0)continue;  
  51.                    int newS = S -(1<<(i-1));  
  52.                     if(dp[newS][j][k] ==-1)continue;  
  53.                     if(!edge[j][k])continue;  
  54.    
  55.                     temp =V[i]+V[i]*V[j]+dp[newS][j][k];  
  56.                     if(edge[i][k])temp +=V[i]*V[j]*V[k];  
  57.                     if(dp[S][i][j] < temp){  
  58.                         dp[S][i][j] = temp;  
  59.                         way[S][i][j] =way[newS][j][k];  //如果dp更新,way直接更新   
  60.                     }  
  61.                     //如果dp不更新,但dp=temp,way加上原来的值   
  62.                     else if(temp ==dp[S][i][j])way[S][i][j] += way[newS][j][k];  
  63.                 }  
  64.            }  
  65.        }  
  66.        long long ans = -1,num = 0;  
  67.        int p = (1<<(N)) - 1;  
  68.        for (int i = 1; i <= N; ++i)  
  69.        for (int j = 1; j <= N; ++j){  
  70.            if(i == j)continue;  
  71.            if (ans < dp[p][i][j]){  
  72.                 ans = dp[p][i][j];  
  73.                 num = way[p][i][j];  
  74.            }  
  75.            else if (ans == dp[p][i][j])  
  76.                 num += way[p][i][j];  
  77.        }  
  78.        if(ans == -1){  
  79.            puts("0 0");  
  80.            continue;  
  81.        }  
  82.        printf("%lld %lld\n",ans,num/2);  
  83.    }  
  84.    return 0;  
  85. }  
#include <cstdio>
#include <cstring>
using namespace std;
const int MAXN = 13;
const int MAX_S = 1<<(MAXN+1);
long long dp[MAX_S][MAXN+1][MAXN+1];
long long way[MAX_S][MAXN+1][MAXN+1];
int edge[MAXN+1][MAXN+1];
long long V[MAXN+1];
int N,M;
int main(){
int cas;
scanf("%d",&cas);
while(cas --){
memset(edge,0,sizeof(edge));
scanf("%d%d",&N,&M);
for(int i = 1; i <= N; ++i)
scanf("%d",&V[i]);
if(N == 1){
printf("%d 1\n",V[1]);
continue;
}
int a,b;
while(M --){
scanf("%d%d",&a,&b);
edge[a][b] = edge[b][a] = 1;
}
memset(dp,-1,sizeof(dp));
memset(way,0,sizeof(way));
int ii,jj;
long long temp;
for(int i = 1; i <= N; ++i)
for(int j = 1; j <= N; ++j){
if(i == j || !edge[i][j])continue;
ii = 1<<(i-1);
jj = 1<<(j-1);
temp = V[i]+V[j]+V[i]*V[j];
dp[ii+jj][i][j] = temp;
way[ii+jj][i][j] = 1;
}
for(int S = 0; S < (1<<N); ++S)
for(int i = 1; i <= N; ++i){
if((S&(1<<(i-1))) == 0)continue;
for(int j = 1; j <= N; ++j){
if((S&(1<<(j-1))) ==0 || i == j || !edge[i][j])continue;
for(int k = 1; k <= N; ++k){
if(i == k || j == k ||(S&(1<<(k-1))) == 0)continue;
int newS = S -(1<<(i-1));
if(dp[newS][j][k] ==-1)continue;
if(!edge[j][k])continue;
temp =V[i]+V[i]*V[j]+dp[newS][j][k];
if(edge[i][k])temp +=V[i]*V[j]*V[k];
if(dp[S][i][j] < temp){
dp[S][i][j] = temp;
way[S][i][j] =way[newS][j][k];  //如果dp更新,way直接更新
}
//如果dp不更新,但dp=temp,way加上原来的值
else if(temp ==dp[S][i][j])way[S][i][j] += way[newS][j][k];
}
}
}
long long ans = -1,num = 0;
int p = (1<<(N)) - 1;
for (int i = 1; i <= N; ++i)
for (int j = 1; j <= N; ++j){
if(i == j)continue;
if (ans < dp[p][i][j]){
ans = dp[p][i][j];
num = way[p][i][j];
}
else if (ans == dp[p][i][j])
num += way[p][i][j];
}
if(ans == -1){
puts("0 0");
continue;
}
printf("%lld %lld\n",ans,num/2);
}
return 0;
}



【ZOJ4257】MostPowerful  

【题目大意】不超过10种气体,两两之间相互碰撞可以产生一定的能量,如a碰b,那么b气体就消失,自身不能碰自身,问最后所能得到的最大能量。

【题目解析】用10位二进制表示气体是否存在,0表示存在,1表示不存在,S(上一个状态)中的两种气体碰撞并且有一种消失,可以得到newS的状态(状态转移)

【状态表示】dp[state] 状态为state时的最大能量

【转移方程】dp[state] = max(dp[state],dp[state']+a[i][j])

【边界条件】dp[i] = 0;

【代码】

[cpp] view plain copy print ?
  1. #include <cstdio>   
  2.    
  3. #include <cstring>   
  4.    
  5. using namespace std;  
  6.    
  7. #define max(a,b) (a) > (b) ? (a) : (b)   
  8.    
  9. const int MAXN = 10;  
  10.    
  11. const int MAX_S = 1 << 10;  
  12.    
  13. int a[MAXN+1][MAXN+1];  
  14.    
  15. int dp[MAX_S];  
  16.    
  17. int N;  
  18.    
  19. int main(){  
  20.    
  21.     while(scanf("%d",&N) != EOF){  
  22.    
  23.         if(N == 0)break;  
  24.    
  25.         for(int i = 0; i < N; ++i)  
  26.    
  27.         for(int j = 0; j < N; ++j){  
  28.    
  29.             scanf("%d",&a[i][j]);  
  30.    
  31.         }  
  32.    
  33.         memset(dp,0,sizeof(dp));  
  34.    
  35.         int full = 1 << N;  
  36.    
  37.         for(int s = 0; s < full; ++s){  
  38.    
  39.             for(int i = 0; i < N; ++i){  
  40.    
  41.                 if((s&(1<<i)))continue;  
  42.    
  43.                 for(int j = 0; j < N; ++j){  
  44.    
  45.                     if(i == j)continue;  
  46.    
  47.                     if( (s&(1<<j)) )continue;  
  48.    
  49.                     int newS = s + (1<<j);  
  50.    
  51.                     dp[newS] = max(dp[newS],dp[s] + a[i][j]);  
  52.    
  53.                 }  
  54.    
  55.    
  56.    
  57.             }  
  58.    
  59.         }  
  60.    
  61.         int ans = 0;  
  62.    
  63.         for(int s = 0; s < full ; ++s)  
  64.    
  65.             ans = max(ans,dp[s]);  
  66.    
  67.         printf("%d\n",ans);  
  68.    
  69.     }  
  70.    
  71.     return 0;  
  72.    
  73. }  
#include <cstdio>
#include <cstring>
using namespace std;
#define max(a,b) (a) > (b) ? (a) : (b)
const int MAXN = 10;
const int MAX_S = 1 << 10;
int a[MAXN+1][MAXN+1];
int dp[MAX_S];
int N;
int main(){
while(scanf("%d",&N) != EOF){
if(N == 0)break;
for(int i = 0; i < N; ++i)
for(int j = 0; j < N; ++j){
scanf("%d",&a[i][j]);
}
memset(dp,0,sizeof(dp));
int full = 1 << N;
for(int s = 0; s < full; ++s){
for(int i = 0; i < N; ++i){
if((s&(1<<i)))continue;
for(int j = 0; j < N; ++j){
if(i == j)continue;
if( (s&(1<<j)) )continue;
int newS = s + (1<<j);
dp[newS] = max(dp[newS],dp[s] + a[i][j]);
}
}
}
int ans = 0;
for(int s = 0; s < full ; ++s)
ans = max(ans,dp[s]);
printf("%d\n",ans);
}
return 0;
}


【POJ2411】Mondriaan'sDream   

【题目大意】一个矩阵,只能放1*2的木块,问将这个矩阵完全覆盖的不同放法有多少种。

【解析】如果是横着的就定义11,如果竖着的定义为竖着的01,这样按行dp只需要考虑两件事儿,当前行&上一行,是不是全为1,不是说明竖着有空(不可能出现竖着的00),另一个要检查当前行里有没有横放的,但为奇数的1。

【状态表示】dp[state][i]第i行状态为state时候的方案数 

【转移方程】dp[state][i] += dp[state'][i-1] state'为i-1行的,不与i行状态state冲突的状态

【边界条件】第一行 符合条件的状态记为1 即dp[state][0] = 1;

【代码】

[cpp] view plain copy print ?
  1. #include <cstdio>   
  2.    
  3. #include <cstring>   
  4.    
  5. using namespace std;  
  6.    
  7. int M,N;  
  8.    
  9. long long dp[1<<11][11];  
  10.    
  11. bool kexing[1<<11];  
  12.    
  13. int full;  
  14.    
  15. inline bool check(int in)  
  16.    
  17. {  
  18.    
  19.         int bit=0;  
  20.    
  21.         while(in>0){  
  22.    
  23.                 if( (in&1)==1)  
  24.    
  25.                          bit++;  
  26.    
  27.                 else{  
  28.    
  29.                          if( (bit&1)==1)  
  30.    
  31.                                   return false;  
  32.    
  33.                          bit=0;  
  34.    
  35.                 }  
  36.    
  37.                 in>>=1;  
  38.    
  39.         }  
  40.    
  41.         if( (bit&1)==1)  
  42.    
  43.                 return false;  
  44.    
  45.         return true;  
  46.    
  47. }  
  48.    
  49. inline bool check2(int x1,int x2){  
  50.    
  51.    if( (x1|x2)!= full-1 )  //如果这里不用位运算,时间很长,要用约3000MS   
  52.    
  53.                 return false;  
  54.    
  55.     return kexing[x1&x2];  
  56.    
  57. }  
  58.    
  59. int main(){  
  60.    
  61.      full = 1<<11;  
  62.    
  63.     for(int s = 0; s < full; ++s)if(check(s))kexing[s] = 1;  
  64.    
  65.     while(scanf("%d%d",&M,&N) != EOF){  
  66.    
  67.         if(N == 0 && M == 0)break;  
  68.    
  69.         memset(dp,0,sizeof(dp));  
  70.    
  71.         full = 1<<N;  
  72.    
  73.         for(int s = 0; s < full; ++s){  
  74.    
  75.             if(kexing[s])  
  76.    
  77.                 dp[s][0] = 1;  
  78.    
  79.         }  
  80.    
  81.         for(int i = 1; i < M; ++i){  
  82.    
  83.             for(int s = 0; s < full; ++s){  
  84.    
  85.                 for(int s1 = 0; s1 < full; ++s1){  
  86.    
  87.                     if(!check2(s1,s))continue;  
  88.    
  89.                     //if(dp[s1][i-1] == 0)continue;     
  90.    
  91.                     /*这一步判断的时间要大于位运算和+=的时间,如果先把这一步放在 
  92.   
  93.                     check2前面,1300多MS,如果放在check2后面,610多MS,如果不加这 
  94.   
  95.                     一步,560MS,但如果check2用的不是位运算,将这一步加在check2前 
  96.   
  97.                     面3000MS左右(水过),如果不加这一步,TLE 
  98.   
  99.                     */  
  100.    
  101.                     dp[s][i] += dp[s1][i-1];  
  102.    
  103.                 }  
  104.    
  105.             }  
  106.    
  107.         }  
  108.    
  109.         int S = (1<<N) - 1;  
  110.    
  111.         printf("%lld\n",dp[S][M-1]);  
  112.    
  113.     }  
  114.    
  115.     return 0;  
  116.    
  117. }  
#include <cstdio>
#include <cstring>
using namespace std;
int M,N;
long long dp[1<<11][11];
bool kexing[1<<11];
int full;
inline bool check(int in)
{
int bit=0;
while(in>0){
if( (in&1)==1)
bit++;
else{
if( (bit&1)==1)
return false;
bit=0;
}
in>>=1;
}
if( (bit&1)==1)
return false;
return true;
}
inline bool check2(int x1,int x2){
if( (x1|x2)!= full-1 )  //如果这里不用位运算,时间很长,要用约3000MS
return false;
return kexing[x1&x2];
}
int main(){
full = 1<<11;
for(int s = 0; s < full; ++s)if(check(s))kexing[s] = 1;
while(scanf("%d%d",&M,&N) != EOF){
if(N == 0 && M == 0)break;
memset(dp,0,sizeof(dp));
full = 1<<N;
for(int s = 0; s < full; ++s){
if(kexing[s])
dp[s][0] = 1;
}
for(int i = 1; i < M; ++i){
for(int s = 0; s < full; ++s){
for(int s1 = 0; s1 < full; ++s1){
if(!check2(s1,s))continue;
//if(dp[s1][i-1] == 0)continue;  
/*这一步判断的时间要大于位运算和+=的时间,如果先把这一步放在
check2前面,1300多MS,如果放在check2后面,610多MS,如果不加这
一步,560MS,但如果check2用的不是位运算,将这一步加在check2前
面3000MS左右(水过),如果不加这一步,TLE
*/
dp[s][i] += dp[s1][i-1];
}
}
}
int S = (1<<N) - 1;
printf("%lld\n",dp[S][M-1]);
}
return 0;
}


【HDU3681】PrisonBreak 状态压缩DP+BFS+二分答案

2010杭州赛区的题目,以现在的水平遇到这种题也就能想一下,赛场上动手写这个题是不会的。前天做状压DP的时候又看到这个题,没想起来怎么做,昨天看了一下解题报告开始下手,遇到了各种问题。调试了N久,终于过了。

【题目大意】机器人从F出发,走到G可以充电,走到Y关掉开关,D不能走进,要求把所有开关关掉,且电量最少,并求出该最小电量。

【题目解析】机器人从出发点出发要求走过所有的Y,因为点很少,所以就能想到经典的TSP问题(起初我也想到了),但关于G点(不要YY)能充电的问题不知道怎么办,看了下解题报告才知道。G点可以充电,到达G点就把当前能量更新为电池容量然后继续走。因为每个G点只能充一次电,这就好像TSP中的每个点只能走一次一样(G和Y都可以走多次,但走到G充电后,该点就变为了S,而走到Y关上开关以后,Y也变成了S。这是一个很巧妙地想法,所以要求Y点只能关一次开关,G点只能充一次电,这就是TSP了。Orz赛场上可以秒杀这题的大神们),然后就是二分答案了,用状压DP判定当前电池容量的情况下是否能符合条件。

【状态表示】dp[s][i]表示到达当前i点状态为s时最大的剩余的能量

【转移方程】同TSP问题了

【边界条件】dp[1<<sid][sid] = rongliang.即出发点的能量就是电池容量

【代码】:

[cpp] view plain copy print ?
  1. #include <cstdio>   
  2.    
  3. #include <cstring>   
  4.    
  5. #include <cmath>   
  6.    
  7. #include <queue>   
  8.    
  9. using namespace std;  
  10.    
  11. #define INF 0x1f1f1f1f   
  12.    
  13. int dp[32769][16];  
  14.    
  15. int dist[16][16][16][16];  
  16.    
  17. int di[4][2] = {{1,0},{-1,0},{0,1},{0,-1}};  
  18.    
  19. int M,N,sid,nCnt,FinalState;  
  20.    
  21. char map[16][16];  
  22.    
  23. struct node{  
  24.    
  25.     int x,y;  
  26.    
  27.     node(){}  
  28.    
  29.     node(int _x,int _y):x(_x),y(_y){}  
  30.    
  31. }nodes[16];  
  32.    
  33.    
  34.    
  35. inline void BFS(node start)  
  36.    
  37. {  
  38.    
  39.     queue<node> que;  
  40.    
  41.     int sx = start.x,sy = start.y;  
  42.    
  43.     dist[sx][sy][sx][sy] = 0;  
  44.    
  45.     que.push(start);  
  46.    
  47.     node cur;  
  48.    
  49.     while(!que.empty()){  
  50.    
  51.         cur = que.front();  
  52.    
  53.         que.pop();  
  54.    
  55.         int x = cur.x,y = cur.y,tx,ty;  
  56.    
  57.         for(int i = 0; i < 4; ++i){  
  58.    
  59.             tx = x + di[i][0];  
  60.    
  61.             ty = y + di[i][1];  
  62.    
  63.             if(tx < 0 || tx >= M || ty < 0 || ty >= N || map[tx][ty] == 'D')continue;  
  64.    
  65.             if(dist[sx][sy][tx][ty] == -1){  
  66.    
  67.                 dist[sx][sy][tx][ty] = dist[sx][sy][x][y] + 1;  
  68.    
  69.                 que.push(node(tx,ty));  
  70.    
  71.             }  
  72.    
  73.         }  
  74.    
  75.     }  
  76.    
  77. }  
  78.    
  79. inline bool ok(int s,int t){ //s状态中走过所有t状态中的城市   
  80.     if(((s&t)&t) == t)return 1;  
  81.     return 0;  
  82. }  
  83. inline bool check(int step){   
  84.     int res = -1;   
  85.     memset(dp,-1,sizeof(dp));   
  86.     dp[1<<sid][sid] = step;   
  87.     int full = 1<<nCnt;   
  88.     for(int s = 0; s < full; ++s){  
  89.    
  90.         for(int i = 0; i < nCnt; ++i){   
  91.             if((s&(1<<i)) == 0 || dp[s][i] == -1)continue;   
  92.             if(ok(s,FinalState))res = max(res,dp[s][i]);   
  93.             for(int j = 0; j < nCnt; ++j){   
  94.                 int temp = dist[nodes[i].x][nodes[i].y][nodes[j].x][nodes[j].y];   
  95.                 if(i == j || temp == -1 || (s&(1<<j)))continue;   
  96.                 temp = dp[s][i] - temp;   
  97.                 if(temp < 0)continue;   
  98.                 int newS = s + (1<<j);   
  99.                 dp[newS][j] = max(dp[newS][j],temp);   
  100.                 if(map[nodes[j].x][nodes[j].y] == 'G')dp[newS][j] = step;  
  101.    
  102.             }  
  103.    
  104.         }  
  105.    
  106.     }  
  107.    
  108.     if(res < 0)return 0;  
  109.    
  110.     return 1;  
  111.    
  112. }  
  113.    
  114.    
  115.    
  116. inline int solve(){  
  117.    
  118.     int low = 0,high = 300;  
  119.    
  120.     int mid;  
  121.    
  122.     while(low <= high){  
  123.    
  124.         mid = (low+high)/2;  
  125.    
  126.         if(check(mid))high = mid-1;  
  127.    
  128.         else low = mid+1;  
  129.    
  130.     }  
  131.    
  132.     if(low == 301)return -1;  
  133.    
  134.     return low;  
  135.    
  136.    
  137.    
  138. }  
  139.    
  140.    
  141.    
  142. int main(){  
  143.    
  144.     while(scanf("%d%d",&M,&N) != EOF){  
  145.    
  146.         if(M == 0 && N == 0)break;  
  147.    
  148.         nCnt = 0;  
  149.    
  150.         FinalState = 0;  
  151.    
  152.         for(int i = 0; i < M; ++i){  
  153.    
  154.             scanf(" %s",map[i]);  
  155.    
  156.             for(int j = 0; j < N; ++j){  
  157.    
  158.                 if(map[i][j] == 'F'){  
  159.    
  160.                     sid = nCnt;  
  161.    
  162.                     nodes[nCnt++] = node(i,j);  
  163.    
  164.                     FinalState += (1<<sid);  
  165.    
  166.                 }  
  167.    
  168.                 else if(map[i][j] == 'G')  
  169.    
  170.                     nodes[nCnt++] = node(i,j);  
  171.    
  172.                 else if(map[i][j] == 'Y'){  
  173.    
  174.                     int tid = nCnt;  
  175.    
  176.                     nodes[nCnt++] = node(i,j);  
  177.    
  178.                     FinalState += (1<<tid);  
  179.    
  180.                 }  
  181.    
  182.             }  
  183.    
  184.         }  
  185.    
  186.         memset(dist,-1,sizeof(dist));  
  187.    
  188.         for(int i = 0; i < nCnt; ++i)BFS(nodes[i]);  
  189.    
  190.         int ans = solve();  
  191.    
  192.         printf("%d\n",ans);  
  193.    
  194.     }  
  195.    
  196.     return 0;  
  197.    
  198. }  
#include <cstdio>
#include <cstring>
#include <cmath>
#include <queue>
using namespace std;
#define INF 0x1f1f1f1f
int dp[32769][16];
int dist[16][16][16][16];
int di[4][2] = {{1,0},{-1,0},{0,1},{0,-1}};
int M,N,sid,nCnt,FinalState;
char map[16][16];
struct node{
int x,y;
node(){}
node(int _x,int _y):x(_x),y(_y){}
}nodes[16];
inline void BFS(node start)
{
queue<node> que;
int sx = start.x,sy = start.y;
dist[sx][sy][sx][sy] = 0;
que.push(start);
node cur;
while(!que.empty()){
cur = que.front();
que.pop();
int x = cur.x,y = cur.y,tx,ty;
for(int i = 0; i < 4; ++i){
tx = x + di[i][0];
ty = y + di[i][1];
if(tx < 0 || tx >= M || ty < 0 || ty >= N || map[tx][ty] == 'D')continue;
if(dist[sx][sy][tx][ty] == -1){
dist[sx][sy][tx][ty] = dist[sx][sy][x][y] + 1;
que.push(node(tx,ty));
}
}
}
}
inline bool ok(int s,int t){ //s状态中走过所有t状态中的城市
if(((s&t)&t) == t)return 1;
return 0;
}
inline bool check(int step){ 
int res = -1; 
memset(dp,-1,sizeof(dp)); 
dp[1<<sid][sid] = step; 
int full = 1<<nCnt; 
for(int s = 0; s < full; ++s){
for(int i = 0; i < nCnt; ++i){ 
if((s&(1<<i)) == 0 || dp[s][i] == -1)continue; 
if(ok(s,FinalState))res = max(res,dp[s][i]); 
for(int j = 0; j < nCnt; ++j){ 
int temp = dist[nodes[i].x][nodes[i].y][nodes[j].x][nodes[j].y]; 
if(i == j || temp == -1 || (s&(1<<j)))continue; 
temp = dp[s][i] - temp; 
if(temp < 0)continue; 
int newS = s + (1<<j); 
dp[newS][j] = max(dp[newS][j],temp); 
if(map[nodes[j].x][nodes[j].y] == 'G')dp[newS][j] = step;
}
}
}
if(res < 0)return 0;
return 1;
}
inline int solve(){
int low = 0,high = 300;
int mid;
while(low <= high){
mid = (low+high)/2;
if(check(mid))high = mid-1;
else low = mid+1;
}
if(low == 301)return -1;
return low;
}
int main(){
while(scanf("%d%d",&M,&N) != EOF){
if(M == 0 && N == 0)break;
nCnt = 0;
FinalState = 0;
for(int i = 0; i < M; ++i){
scanf(" %s",map[i]);
for(int j = 0; j < N; ++j){
if(map[i][j] == 'F'){
sid = nCnt;
nodes[nCnt++] = node(i,j);
FinalState += (1<<sid);
}
else if(map[i][j] == 'G')
nodes[nCnt++] = node(i,j);
else if(map[i][j] == 'Y'){
int tid = nCnt;
nodes[nCnt++] = node(i,j);
FinalState += (1<<tid);
}
}
}
memset(dist,-1,sizeof(dist));
for(int i = 0; i < nCnt; ++i)BFS(nodes[i]);
int ans = solve();
printf("%d\n",ans);
}
return 0;
}


这篇关于状态压缩DP总结(谨记大牛总结)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/978379

相关文章

Python中实现进度条的多种方法总结

《Python中实现进度条的多种方法总结》在Python编程中,进度条是一个非常有用的功能,它能让用户直观地了解任务的进度,提升用户体验,本文将介绍几种在Python中实现进度条的常用方法,并通过代码... 目录一、简单的打印方式二、使用tqdm库三、使用alive-progress库四、使用progres

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

Java向kettle8.0传递参数的方式总结

《Java向kettle8.0传递参数的方式总结》介绍了如何在Kettle中传递参数到转换和作业中,包括设置全局properties、使用TransMeta和JobMeta的parameterValu... 目录1.传递参数到转换中2.传递参数到作业中总结1.传递参数到转换中1.1. 通过设置Trans的

C# Task Cancellation使用总结

《C#TaskCancellation使用总结》本文主要介绍了在使用CancellationTokenSource取消任务时的行为,以及如何使用Task的ContinueWith方法来处理任务的延... 目录C# Task Cancellation总结1、调用cancellationTokenSource.

Qt实现文件的压缩和解压缩操作

《Qt实现文件的压缩和解压缩操作》这篇文章主要为大家详细介绍了如何使用Qt库中的QZipReader和QZipWriter实现文件的压缩和解压缩功能,文中的示例代码简洁易懂,需要的可以参考一下... 目录一、实现方式二、具体步骤1、在.pro文件中添加模块gui-private2、通过QObject方式创建

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

hdu1565(状态压缩)

本人第一道ac的状态压缩dp,这题的数据非常水,很容易过 题意:在n*n的矩阵中选数字使得不存在任意两个数字相邻,求最大值 解题思路: 一、因为在1<<20中有很多状态是无效的,所以第一步是选择有效状态,存到cnt[]数组中 二、dp[i][j]表示到第i行的状态cnt[j]所能得到的最大值,状态转移方程dp[i][j] = max(dp[i][j],dp[i-1][k]) ,其中k满足c

hdu4826(三维DP)

这是一个百度之星的资格赛第四题 题目链接:http://acm.hdu.edu.cn/contests/contest_showproblem.php?pid=1004&cid=500 题意:从左上角的点到右上角的点,每个点只能走一遍,走的方向有三个:向上,向下,向右,求最大值。 咋一看像搜索题,先暴搜,TLE,然后剪枝,还是TLE.然后我就改方法,用DP来做,这题和普通dp相比,多个个向上