传送门

题目大意:

给两个数组, 数组中的两个元素可以合并成两元素之和,每个元素都可以分裂成相应的大小,问从数组1变化到数组2至少需要多少步?

题目分析:

看到数据范围\(n<=10\), 显然是在提醒我们状压。用sum[i]表示i状态的面积总和。

对于任意一个数组\(1\)和数组\(2\)(\(sum\)值必须相等才能变化),变化的最劣情况是将数组\(1\)合并成1个再分成数组\(2\),步数为\(n1 + n2 - 2\),然而如果数组\(1\)的一个子集\(k\)和数组\(2\)的一个子集\(l\)的\(sum\)相等,\(k\)和\(l\)又可以以同样的方法进行变化,使数组\(1\)到数组\(2\)的变化次数\(-2 = n1 + n2 - 2 * 2\),继续拓展,也就是说如果存在\(x\)个面积相等的子集,变化次数就为\(n1 + n2 - 2 * x\)。

\(f[i][j]\)表示数组\(1\)的\(i\)状态和数组\(2\)的\(j\)状态存在的相同面积的子集数量,于是就可以枚举子集进行\(dp\)求出\(f[(1 << n1) - 1][(1 << n2) - 1]\),最后答案即为\(n1 + n2 - 2 * f[(1 << n1) - 1][(1 << n2) - 1]\)。

看似复杂度会爆表,其实无用状态很多,因为只有\(sum\)相等才能进行变化。

code

580ms

#include<bits/stdc++.h>
using namespace std; const int N = 15;
int n1, n2, a1[N], a2[N], sum1[1100], sum2[1100], f[1100][1100]; inline int count(int x){
int ret = 0;
while(x) ret++, x = x & (x - 1);
return ret;
} struct ioSys{
inline int read(){
int i = 0, f = 1; char ch = getchar();
for(; (ch < '0' || ch > '9') && ch != '-'; ch = getchar());
if(ch == '-') f = -1, ch = getchar();
for(; ch >= '0' && ch <= '9'; ch = getchar())
i = (i << 3) + (i << 1) + (ch - '0');
return i * f;
} inline void wr(int x){
if(x < 0) putchar('-'), x = -x;
if(x > 9) wr(x / 10);
putchar(x % 10 + '0');
}
inline void operator >> (int &x){
x = read();
}
inline void operator << (int x){
wr(x);
}
}IO; int main(){
freopen("h.in", "r", stdin);
IO >> n1;
for(int i = 1; i <= n1; i++) IO >> a1[i];
for(int i = 0; i <= (1 << n1) - 1; i++){
for(int j = 1; j <= n1; j++)
if((1 << (j - 1)) & i) sum1[i] += a1[j];
// cout<<sum1[i]<<" ";
}
// cout<<endl;
IO >> n2;
for(int i = 1; i <= n2; i++) IO >> a2[i];
for(int i = 0; i <= (1 << n2) - 1; i++){
for(int j = 1; j <= n2; j++)
if((1 << (j - 1)) & i) sum2[i] += a2[j];
// cout<<sum2[i]<<" ";
}
// cout<<endl;
//---------------------------------------
for(int i = 0; i <= (1 << n1) - 1; i++)
for(int j = 0; j <= (1 << n2) - 1; j++){
if(sum1[i] == sum2[j]){
f[i][j]++;
for(int k = i & (i - 1); k; k = i & (k - 1))
for(int l = j & (j - 1); l; l = j & (l - 1)){
// if(sum1[k] == sum2[l]) f[k][l]++;
f[i][j] = max(f[i][j], f[i - k][j - l] + f[k][l]);
}
}
}
// cout<<f[(1 << n1) - 1][(1 << n2) - 1]<<endl;
IO << n1 + n2 - 2 * f[(1 << n1) - 1][(1 << n2) - 1];
}

BZOJ 2064 - 状压DP的更多相关文章

  1. bzoj 1879 状压dp

    879: [Sdoi2009]Bill的挑战 Time Limit: 4 Sec  Memory Limit: 64 MBSubmit: 852  Solved: 435[Submit][Status ...

  2. bzoj 1087 状压dp

    1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4130  Solved: 2390[Submit][ ...

  3. BZOJ 4057 状压DP

    思路: 状压一下 就完了... f[i]表示选了的集合为i 转移的时候判一判就好了.. //By SiriusRen #include <cstdio> #include <cstr ...

  4. BZOJ 4565 状压DP

    思路: f[i][j][S]表示从i到j压成S状态 j-m是k-1的倍数 $f[i][j][S<<1]=max(f[i][j][S<<1],f[i][m-1][S]+f[m][ ...

  5. bzoj 1072状压DP

    1072: [SCOI2007]排列perm Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2293  Solved: 1448[Submit][St ...

  6. bzoj 1072 状压DP

    我们用w[i][j]来表示,i是一个二进制表示我们选取了s中的某些位,j表示这些位%d为j,w[i][j]则表示这样情况下的方案数,那么我们可以得到转移.w[i|(1<<k)][(j*10 ...

  7. bzoj 2669 状压DP

    因为最多有8个'X',所以我们可以用w[i][s]来表示现在我们填了前i个数,填的X的为S,因为每次新加进来的数都不影响前面的最小值,所以我们可以随便添加,这样就有了剩下所有位置的方案,每次都这样转移 ...

  8. bzoj 1076 状压DP

    我们设w[i][s]为当前到第i关,手中的物品为s的时候,期望得分为多少,其中s为二进制表示每种物品是否存在. 那么就比较容易转移了w[i][s]=(w[i-1][s']+v[j]) *(1/k),其 ...

  9. BZOJ 1231 状压DP

    思路: f[i][j] i表示集合的组成 j表示选最后一个数 f[i][j]表示能选的方案数 f[i|(1<< k)][k]+=f[i][j]; k不属于i j属于i且符合题意 最后Σf[ ...

随机推荐

  1. 洛谷 P1657 选书

    P1657 选书 题目描述 学校放寒假时,信息学奥赛辅导老师有1,2,3……x本书,要分给参加培训的x个人,每人只能选一本书,但是每人有两本喜欢的书.老师事先让每个人将自己喜欢的书填写在一张表上.然后 ...

  2. amazeui学习笔记--css(常用组件11)--分页Pagination

    amazeui学习笔记--css(常用组件11)--分页Pagination 一.总结 1.分页使用:还是ul包li的形式: 分页组件,<ul> / <ol> 添加 .am-p ...

  3. node event中 on emit off 的封装

    事件绑定一个事件名称对应多个事件函数 应此它们的关系是一对多的关系 数据类型采用对象的形式 key:val 因为函数有多个 所以val选用数组 事件仓库 eventList = { key:val, ...

  4. sql语句的编程手册 SQL PLUS

    一.SQL PLUS 引言 SQL命令 以下17个是作为语句开头的关键字: alter drop revoke audit grant rollback* commit* insert select ...

  5. GO语言学习(一)Windows 平台下 Go 语言的安装和环境变量设置

    1. Go 语言 SDK 安装包下载和安装 GO语言安装包下载地址:https://www.golangtc.com/download 下载 go1.9.2.windows-amd64 2. Go 语 ...

  6. PHP Web木马扫描器代码

    <?php header('content-type:text/html;charset=gbk'); set_time_limit(0);//防止超时 /** * * php目录扫描监控增强版 ...

  7. C语言free函数的原理——————————【Badboy】

    今天在网上看到了这样一个问题,"假设malloc 了一块字符串的内存.然后,它改变了这个字符串的大小,问会不会有一部分内存没有被释放掉."这个问题,曾经的确没有细致想过. 当然.我 ...

  8. thinkphp自动完成、软删除 和时间戳

    thinkphp自动完成.软删除 和时间戳 一.总结 自动完成:没有手动赋值的情况下进行手动处理 软删除:实现假删除,可以进行恢复 时间戳:系统支持自动写入创建和更新的时间戳字段 二.thinkphp ...

  9. Linux中U盘和SD卡加载卸载命令

    U盘挂载命令U盘插入的时候会显示启动信息,启动信息中sda: sda1指U盘的设备名为sda1dev设备目录下有一个sda1设备文件,此设备文件就是我们插入的U盘,我们将这个设备文件挂载到Linux系 ...

  10. [转]在Win7 64位注册ActiveX控件

    原文作者: K_H_H 原文地址: http://blog.sina.com.cn/s/blog_56b96c5a0100ztc7.html  首先必须以管理员身份运行cmd.exe,即在cmd.ex ...