题目链接:hdu 4778 Rabbit Kingdom

题目大意:Alice和Bob玩游戏,有一个炉子。能够将S个同样颜色的宝石换成一个魔法石。如今有B个包,每一个包里有若干个宝石,给出宝石的颜色。如今由Alice開始,两人轮流选取一个包的宝石放入炉中,每当获得一个魔法石时,能够额外获得一次机会再选一个包放入。两人均依照自己的最优策略。问说最后Alice的魔法石-Bob的魔法石是多少。

解题思路:状态压缩,221,对于每次移动到下一个状态,假设获得的魔法石g非零。则说明下一个状态还是自己在取。则要选择最优的。假设g为0。则说明下一个状态不是自己在取,则要取尽量小的,相应也就是相反数尽量大的。

C++ 记忆化版
#include <cstdio>
#include <cstring>
#include <algorithm> using namespace std; const int maxb = 21;
const int maxn = 1<<21;
const int maxg = 10;
const int INF = 0x3f3f3f3f; int G, B, S;
int c[maxb+5][maxg], s[maxn+5][maxg];
int v[maxn+5], dp[maxn+5]; void init () {
int t, a;
memset(c, 0, sizeof(c));
memset(v, 0, sizeof(v)); for (int i = 0; i < B; i++) {
scanf("%d", &t);
for (int j = 0; j < t; j++) {
scanf("%d", &a);
c[i][a]++;
}
} for (int i = 0; i < (1<<B); i++) {
for (int j = 0; j < B; j++) { if (i&(1<<j))
continue; int e = i|(1<<j); if (v[e])
continue; for (int k = 1; k <= G; k++)
s[e][k] = (s[i][k] + c[j][k]) % S;
}
}
} int add (int k, int x) {
int ans = 0;
for (int i = 1; i <= G; i++)
ans += (s[k][i] + c[x][i]) / S;
return ans;
} int dfs (int u) { if (u + 1 == (1<<B))
return 0; if (v[u])
return dp[u]; int up = -INF;
int lower = INF; for (int i = 0; i < B; i++) {
if (u&(1<<i))
continue; int g = add (u, i); if (g)
up = max(up, dfs(u|(1<<i))+g);
else
lower = min(lower, dfs(u|1<<i));
}
v[u] = 1;
return dp[u] = max(up, -lower);
} int main () {
while (scanf("%d%d%d", &G, &B, &S) == 3 && G + B + S) {
init();
memset(v, 0, sizeof(v));
printf("%d\n", dfs(0));
}
return 0;
}
C++ 递推版
#include <cstdio>
#include <cstring>
#include <algorithm> using namespace std; const int maxb = 21;
const int maxn = 1<<21;
const int maxg = 10;
const int INF = 0x3f3f3f3f; int G, B, S;
int c[maxb+5][maxg], s[maxn+5][maxg];
int v[maxn+5], dp[maxn+5]; void init () {
int t, a;
memset(c, 0, sizeof(c));
memset(v, 0, sizeof(v)); for (int i = 0; i < B; i++) {
scanf("%d", &t);
for (int j = 0; j < t; j++) {
scanf("%d", &a);
c[i][a]++;
}
} for (int i = 0; i < (1<<B); i++) {
for (int j = 0; j < B; j++) { if (i&(1<<j))
continue; int e = i|(1<<j); if (v[e])
continue; for (int k = 1; k <= G; k++)
s[e][k] = (s[i][k] + c[j][k]) % S;
}
}
} int add (int k, int x) {
int ans = 0;
for (int i = 1; i <= G; i++)
ans += (s[k][i] + c[x][i]) / S;
return ans;
} int solve () {
int e = (1<<B) - 1;
memset(dp, -INF, sizeof(dp));
dp[e] = 0; for (int u = e; u >= 0; u--) { for (int i = 0; i < B; i++) {
if ((u&(1<<i)) == 0)
continue; int ui = u-(1<<i);
int g = add(ui, i); if (g)
dp[ui] = max(dp[ui], dp[u] + g);
else
dp[ui] = max(dp[ui], -dp[u]);
}
}
return dp[0];
} int main () {
while (scanf("%d%d%d", &G, &B, &S) == 3 && G + B + S) {
init();
printf("%d\n", solve());
}
return 0;
}

版权声明:本文博客原创文章,博客,未经同意,不得转载。

hdu 4778 Rabbit Kingdom(减少国家)的更多相关文章

  1. HDU 4777 Rabbit Kingdom(树状数组)

    HDU 4777 Rabbit Kingdom 题目链接 题意:给定一些序列.每次询问一个区间,求出这个区间和其它数字都互质的数的个数 #include <cstdio> #include ...

  2. HDU 4777 Rabbit Kingdom (2013杭州赛区1008题,预处理,树状数组)

    Rabbit Kingdom Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)To ...

  3. HDU 4777 Rabbit Kingdom --容斥原理+树状数组

    题意: 给一个数的序列,询问一些区间,问区间内与区间其他所有的数都互质的数有多少个. 解法: 直接搞有点难, 所谓正难则反,我们求区间内与其他随便某个数不互质的数有多少个,然后区间长度减去它就是答案了 ...

  4. HDU 4777 Rabbit Kingdom 树状数组

    分析:找到每一个点的左边离他最近的不互质数,记录下标(L数组),右边一样如此(R数组),预处理 这个过程需要分解质因数O(n*sqrt(n)) 然后离线,按照区间右端点排序 然后扫一遍,对于当前拍好顺 ...

  5. HDU 4777 Rabbit Kingdom

    素因子分解,树状数组.$ACM/ICPC$ $2013$杭州区域赛$H$题. 首先需要处理出数字$a[i]$左边最远到$L[i]$,右边最远到$R[i]$区间内所有数字都与$a[i]$互质. 那么对于 ...

  6. hdu 4778 Gems Fight! 博弈+状态dp+搜索

    作者:jostree 转载请注明出处 http://www.cnblogs.com/jostree/p/4102743.html 题目链接:hdu 4778 Gems Fight! 博弈+状态dp+搜 ...

  7. hdu 5030 Rabbit&#39;s String(后缀数组&amp;二分法)

    Rabbit's String Time Limit: 40000/20000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others ...

  8. HDU 4778 状压DP

    一看就是状压,由于是类似博弈的游戏.游戏里的两人都是绝对聪明,那么先手的选择是能够确定最终局面的. 实际上是枚举最终局面情况,0代表是被Bob拿走的,1为Alice拿走的,当时Alice拿走且满足变换 ...

  9. hdu 4778 Gems Fight! 状压dp

    转自wdd :http://blog.csdn.net/u010535824/article/details/38540835 题目链接:hdu 4778 状压DP 用DP[i]表示从i状态选到结束得 ...

随机推荐

  1. Android 代码设置密码输入框内容的显示/隐藏

    //内容可见 mEtPassword.setTransformationMethod(HideReturnsTransformationMethod.getInstance()); //内容不可见 m ...

  2. nginx记录响应与POST请求日志

    生产环境中的某些api出现故障,但是问题无法重现,但是又很想解决掉问题以及我们新项目上线,需要跟踪请求与响应的信息,可以预先找到一些bug,减少大面积的损失. 安装nginx与ngx_lua 响应日志 ...

  3. SQLite数据转换成sql server数据

    需要将SQLite数据库里的数据导入到SQL Server,在网上搜了好久,没有找到一个方便实用的方法. 经过本人的细心琢磨实验,终于让我给找到一好的方法:使用CSV文件作为介质来做转换.现在记录下来 ...

  4. 最简单也最难——如何获取到Android控件的高度

    问题 如何获取一个控件的长和高,相信很多朋友第一眼看见这个问题都会觉得很简单,直接在onCreate里面调用getWidth.getMeasuredWidth不就可以获得了吗,但是,事实上是并没有简单 ...

  5. urlrewrite 匹配规则之优先选择

    urlrewrite rule可以使用java的正则表达式匹配规则,但是这里存在一个问题点,假如有一个通配的规则和一个精确匹配的规则,urlrewrite 会选择那个去匹配呢? 如下两种规则: < ...

  6. ASP.NET Excel导入到SQL Server数据库

    本文转自:http://www.cnblogs.com/lhking/archive/2009/06/08/1499002.html 提供把Excel里的数据导入到SQL Server 数据库,前提是 ...

  7. 目录 of 2013-2014-1(内容已更新结束)

    (内容已更新结束) UML部分: ---------------1.概述2.用例图3.类图4.顺序图 MVC部分: ----------------1.概述2.路由3.控制器4.视图5.模型6.安装部 ...

  8. httplib,urllib和urllib2

    一.httplib实现了HTTP和HTTPS的客户端协议,一般不直接使用,在python更高层的封装模块中(urllib,urllib2)使用了它的http实现. import httplib con ...

  9. 盘点:#AzureChat - 虚拟机和自动伸缩

    感谢大家跟 Corey Sanders 和 Stephen Siciliano 一起参加本次 #AzureChat.我们很高兴能借这次在线讨论的机会,倾听各位社区成员对我们最受欢迎的两个主题的意见 - ...

  10. Web网页中内嵌Activex的Activex插件开发 .

    转载自: http://blog.csdn.net/tttyd/article/details/5258096 源代码下载 http://files.cnblogs.com/tttyd/Activex ...