题意:就是给了你一些货物的重量,然后给了两辆车一次的载重,让你求出最少的运输次数。

分析:首先要从一辆车入手,搜出所有的一次能够运的所有状态,然后把两辆车的状态进行合并,最后就是解决了,有两种方法:

1.组合解决:

代码实现:

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std; int n,sum1,sum2,a[];
int st1[],st2[],st[],num1,num2,num;
int total[],all;
//st1数组保存的是第一辆车能够一次运走的所有状态,同理st2数组为第二辆车的
//st数组保存的是两辆车一次能够运走的所有状态
void dfs(int f,int flag,int x,int s)//开始的时候这个dfs是作死的错,说明dfs写得还不够熟练
{
int i,j,temp;
if(flag==&&f==n+)
st1[num1++]=x;
else if(f==n+)
st2[num2++]=x;
if(f>n) return ;
temp=<<(n-f);
if(flag==)
{
if(s+a[f]<=sum1)
dfs(f+,flag,x+temp,s+a[f]);
dfs(f+,flag,x,s);
}
else
{
if(s+a[f]<=sum2)
dfs(f+,flag,x+temp,s+a[f]);
dfs(f+,flag,x,s);
}
} void hebing()//两辆车的状态合并
{
int i,j,t=;
int visited[],temp;
memset(visited,,sizeof(visited));
for(i=;i<num1;i++)
for(j=;j<num2;j++)
{
temp=st1[i]|st2[j];
if(visited[temp]==)
{
st[num++]=temp;
visited[temp]=;
}
}
} void solve(int T)//我这里是用组合解决的,虽然提交了之后发现用组合比用背包时间还少
{ //但是觉得可能是测试数据的原因,个人觉得还是背包靠谱些
printf("Scenario #%d:\n",T);
int temp[],t,res=,x;
int i,j,max=,flag=;
int visited[];
all=;total[]=;
for(i=;i<=n;i++)
max=max+(<<(i-));
while()
{
res++;t=;
memset(visited,,sizeof(visited));
for(i=;i<all;i++)
{
// int kao=0;
//kao++;
for(j=;j<num;j++)
{
x=(total[i]|st[j]);
if(x==max)
{
flag=;
break;
}
if(visited[x]==)
{
visited[x]=;
temp[t++]=x;
}
}
if(flag==)
break;
}
if(flag==)
break;
for(i=;i<t;i++)
total[i]=temp[i];
all=t;
}
printf("%d\n",res);
} int main()
{
int i,T,t;
scanf("%d",&T);
for(t=;t<=T;t++)
{
num1=;
num2=;
num=;
scanf("%d%d%d",&n,&sum1,&sum2);
for(i=; i<=n; i++)
scanf("%d",&a[i]);
dfs(,,,);
dfs(,,,);
hebing();
solve(t);
if(t!=T)
printf("\n");
}
return ;
}

2.背包解决:
代码实现:

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std; int n,sum1,sum2,a[];
int st1[],st2[],st[],num1,num2,num;
int total[],all; void dfs(int f,int flag,int x,int s)
{
int i,j,temp;
if(flag==&&f==n+)
st1[num1++]=x;
else if(f==n+)
st2[num2++]=x;
if(f>n) return ;
temp=<<(n-f);
if(flag==)
{
if(s+a[f]<=sum1)
dfs(f+,flag,x+temp,s+a[f]);
dfs(f+,flag,x,s);
}
else
{
if(s+a[f]<=sum2)
dfs(f+,flag,x+temp,s+a[f]);
dfs(f+,flag,x,s);
}
} void hebing()
{
int i,j,t=;
int visited[],temp;
memset(visited,,sizeof(visited));
for(i=;i<num1;i++)
for(j=;j<num2;j++)
{
temp=st1[i]|st2[j];
if(visited[temp]==)
{
st[num++]=temp;
visited[temp]=;
}
}
} int Min(int x,int y)
{
return x>y?y:x;
} void solve(int T)
{
printf("Scenario #%d:\n",T);
int i,j,dp[];
for(i=;i<(<<n);i++)
dp[i]=;
dp[]=;
for(i=;i<num;i++)
{
for(j=(<<n)-;j>=;j--)
{
if(dp[j]==)
continue;
if((j&st[i])==)
dp[j|st[i]]=Min(dp[j|st[i]],dp[j]+);
}
}
printf("%d\n",dp[(<<n)-]);
} int main()
{
int i,T,t;
scanf("%d",&T);
for(t=;t<=T;t++)
{
num1=;
num2=;
num=;
scanf("%d%d%d",&n,&sum1,&sum2);
for(i=; i<=n; i++)
scanf("%d",&a[i]);
dfs(,,,);
dfs(,,,);
hebing();
solve(t);
if(t!=T)
printf("\n");
}
return ;
}

poj 2923(状态压缩dp)的更多相关文章

  1. POJ 1185 状态压缩DP(转)

    1. 为何状态压缩: 棋盘规模为n*m,且m≤10,如果用一个int表示一行上棋子的状态,足以表示m≤10所要求的范围.故想到用int s[num].至于开多大的数组,可以自己用DFS搜索试试看:也可 ...

  2. POJ 1185 状态压缩DP 炮兵阵地

    题目直达车:   POJ 1185 炮兵阵地 分析: 列( <=10 )的数据比较小, 一般会想到状压DP. Ⅰ.如果一行10全个‘P’,满足题意的状态不超过60种(可手动枚举). Ⅱ.用DFS ...

  3. poj 2688 状态压缩dp解tsp

    题意: 裸的tsp. 分析: 用bfs求出随意两点之间的距离后能够暴搜也能够用next_permutation水,但效率肯定不如状压dp.dp[s][u]表示从0出发訪问过s集合中的点.眼下在点u走过 ...

  4. Mondriaan's Dream(POJ 2411状态压缩dp)

    题意:用1*2的方格填充m*n的方格不能重叠,问有多少种填充方法 分析:dp[i][j]表示i行状态为j时的方案数,对于j,0表示该列竖放(影响下一行的该列),1表示横放成功(影响下一列)或上一列竖放 ...

  5. poj 2411 状态压缩dp

    思路:将每一行看做一个二进制位,那么所有的合法状态为相邻为1的个数一定要为偶数个.这样就可以先把所有的合法状态找到.由于没一层的合法状态都是一样的,那么可以用一个数组保存.由第i-1行到第i行的状态转 ...

  6. poj 3254 状态压缩DP

    思路:把每行的数当做是一个二进制串,0不变,1变或不变,找出所有的合法二进制形式表示的整数,即相邻不同为1,那么第i-1行与第i行的状态转移方程为dp[i][j]+=dp[i-1][k]: 这个方程得 ...

  7. POJ 3254 状态压缩 DP

    B - Corn Fields Crawling in process... Crawling failed Time Limit:2000MS     Memory Limit:65536KB    ...

  8. poj 2923(状态压缩+背包)

    比较巧妙的一道题目,拿到题目就想用暴力直接搜索,仔细分析了下发现复杂度达到了2^n*n! ,明显不行,于是只好往背包上想. 于是又想二分找次数判断可行的方法,但是发现复杂度10^8还是很悬... 然后 ...

  9. POJ 3691 (AC自动机+状态压缩DP)

    题目链接:  http://poj.org/problem?id=3691 题目大意:给定N个致病DNA片段以及一个最终DNA片段.问最终DNA片段最少修改多少个字符,使得不包含任一致病DNA. 解题 ...

随机推荐

  1. 【重走Android之路】【路线篇(二)】知识点归纳

    [重走Android之路][路线篇(二)]知识点归纳   参考:http://blog.csdn.net/xujing81/article/details/7313507   第一阶段:Java面向对 ...

  2. c# 计算一个整型数组的平均

    一个整型数组的平均: class Program { static void Main(string[] args) { ,,,,,,,,,}; double avg= GetAvg(array); ...

  3. [HDOJ2196]Computer (树直径, 树DP)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2196 给一棵树,求树上各点到某点的距离中最长的距离.注意每个点都要求. 和普通求树的直径不一样,要求每 ...

  4. Webservce、WCF、WebApi的区别

    Web Service It is based on SOAP and return data in XML form. It support only HTTP protocol. It is no ...

  5. 【转】Android Launcher研究 (一)

    这份源码是基于2.1的launcher2,以后版本虽有变化,但大概的原理一直还是保留了. 一.主要文件和类  1.Launcher.java:launcher中主要的activity. 2.DragL ...

  6. Codeforces Round #362 (Div. 2) A.B.C

    A. Pineapple Incident time limit per test 1 second memory limit per test 256 megabytes input standar ...

  7. 浅谈javascript中的作用域

    首先说明一下:Js中的作用域不同于其他语言的作用域,要特别注意     JS中作用域的概念: 表示变量或函数起作用的区域,指代了它们在什么样的上下文中执行,亦即上下文执行环境.Javascript的作 ...

  8. 基于Bootstrap的超酷jQuery开关按钮插件

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAJgAAAA2CAIAAAC0tsfoAAAB/0lEQVR4nO3bQWrCQBTGcc/mbVz0DD

  9. 采用RPC方式和document方式 开发Axis2的WebService客户端

    import javax.xml.namespace.QName; import org.apache.axiom.om.OMAbstractFactory; import org.apache.ax ...

  10. LA 2965 Jurassic Remains

    这是我做的第一道状态压缩的题目,而且我自己居然看懂了,理解得还算透彻. 题意:给出若干个大写字母组成的字符串,然后选取尽量多的字符串使得这些字母出现偶数次. 最朴素的想法,穷举法:每个字符串只有选和不 ...