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

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

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. Linux command: usermod -- 改变用户状态

    应用举例: 1. usermod -g newuser newuser force use GROUP as new primary group. 一般时候是默认的,也是必须的(不能更改).2. 指定 ...

  2. 如何计算一个字符串表示的计算式的值?——C_递归算法实现

    在<C程序设计伴侣>的8.7.3 向main()函数传递数据这一小节中,我们介绍了如何通过main()函数的参数,向程序传递两个数据并计算其和值的简单加法计算器add.exe.这个程序,好 ...

  3. 基于RPC原理的dubbo

    在校期间大家都写过不少程序,比如写个hello world服务类,然后本地调用下,如下所示.这些程序的特点是服务消费方和服务提供方是本地调用关系. 而一旦踏入公司尤其是大型互联网公司就会发现,公司的系 ...

  4. React-非dom属性-ref标签

    <!DOCTYPE html> <html lang="zh-cn"> <head> <meta charset="UTF-8& ...

  5. vmware通过vmnet8共享本地网络

    转载于:http://zhaolongchn.blog.163.com/blog/static/19065850420122595117886/ 1,首先将真实电脑的虚拟网卡VMnet8启用 2,然后 ...

  6. Centos环境下部署游戏服务器-Eclipse

    一直在想这篇文章该不该写,因为这篇文章更像是教你如何使用一个ide这种文章,毫无价值可言.但思来想去还是应给写.上篇文章主要说了编译原理和过程,这篇文章就是理论联系实际的典范.并且很多工程师一辈子都不 ...

  7. Largest Rectangle in Histogram-最大长方形

    题目描述: 给定n个非负整数height[n],分别代表直方图条的高,每个条的宽设为1,求直方图中面积最大的矩形的面积 题目来源: http://oj.leetcode.com/problems/la ...

  8. JVM内存区域划分

    前言 Java程序的运行是通过Java虚拟机来实现的.通过类加载器将class字节码文件加载进JVM,然后根据预定的规则执行.Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同 ...

  9. Java基本开发环境搭建

    转http://blog.csdn.net/cxwen78/article/details/6400798 eclipse怎么编写JAVA,然后运行 1.新建一个Java project取个名字(首字 ...

  10. OpenGL基础渲染

    客户端-服务器 客户端是存储在CPU存储器中的,并且在应用程序中执行(或者驱动程序),驱动程序将渲染命令和数据组合起来,发动到服务器执行.服务器和客户机在功能上是异步的,他们是各自独立的软件模块或者硬 ...