四维dp 或者 剪枝 + dfs Codeforces Beta Round #6 (Div. 2 Only) D
http://codeforces.com/contest/6/problem/D
题目大意:有一队人,排成一列,每个人都有生命值,你每次可以攻击2~n位置的一个的人,假设每次攻击的位置为pos,那么pos位受到a点伤害,pos-1和pos+1受到b点伤害。问让所有人生命值都小于0所需要的最少操作数。
思路:最近在加强dp类型的题目,但是因为做过一点IDA*,感觉这题貌似可以用IDA*差不多的思路来做哈?(关键是我不会DP,TAT)然后我们找寻一下IDA*的可行性剪枝就好了:目前数列的maxval<=(lim - deep) * a即可,否则TLE了
这是IDA*的代码
//看看会不会爆int!数组会不会少了一维!
//取物问题一定要小心先手胜利的条件
#include <bits/stdc++.h>
using namespace std;
#define LL long long
#define ALL(a) a.begin(), a.end()
#define pb push_back
#define mk make_pair
#define fi first
#define se second
#define haha printf("haha\n")
const int maxn = ;
const int inf = 0x3f3f3f3f;
int road[maxn * maxn];
int n, a, b;
int h[maxn]; bool dfs(int pos, int deep, int lim){ int maxcnt = -inf;
for (int i = ; i <= n; i++) maxcnt = max(maxcnt, h[i]);
if (deep > lim) return false;
if (maxcnt > (lim - deep) * a) return false;//加上这个剪枝就变得超快
if (deep == lim){
if (maxcnt >= )return false;
else return true;
}
if (pos == n) return false; int cnt = max(h[pos - ] / b, max(h[pos] / a, h[pos + ] / b)) + ;
for (int cnt1 = ; cnt1 <= cnt; cnt1++){
if (h[pos - ] >= cnt1 * b) continue;
h[pos - ] -= b * cnt1; h[pos] -= a * cnt1; h[pos + ] -= b * cnt1;
for (int j = deep + ; j <= deep + cnt1; j++)
road[j] = pos;
if (dfs(pos + , deep + cnt1, lim)) {
for (int j = deep + ; j <= deep + cnt1; j++) road[j] = pos;
return true;
}
h[pos - ] += b * cnt1; h[pos] += a * cnt1; h[pos + ] += b * cnt1;
}
return false;
} int main(){
memset(road, , sizeof(road));
cin >> n >> a >> b;
int beg = ;
for (int i = ; i <= n; i++) scanf("%d", h + i);
for (int i = ; i <= * ; i++)
if (dfs(, , i)){
printf("%d\n", i);
for (int j = ; j <= i; j++)
printf("%d%c", road[j], j == i ? '\n' : ' ');
break;
}
return ;
}
明天附上DP代码。晚安~
dp思路:
定义dp[i][j][k][z]表示目前攻击第i个人,j表示第i-1个人的生命,k表示第i个人的生命,z表示第i+1个人的生命。然后转移需要注意一下以下方面
①简单的转移:dp[i][j - b][k - a][z - b] = dp[i][j][k][z];
②如果j-b以后的生命值为0了,那么dp[i+1][k - a][z - b][h[i + 2]]=dp[i][j][k][z] + 1;
③如果j本来就为0,那么dp[i+1][k][z][h[i + 2]] = dp[i][j][k][z];
//看看会不会爆int!数组会不会少了一维!
//取物问题一定要小心先手胜利的条件
#include <bits/stdc++.h>
using namespace std;
#define LL long long
#define ALL(a) a.begin(), a.end()
#define pb push_back
#define mk make_pair
#define fi first
#define se second
#define haha printf("haha\n")
const int maxn = ;
const int inf = 0x3f3f3f3f;
struct Point{
int i, j, k, z;
Point(int i = , int j = , int k = , int z = ): i(i), j(j), k(k), z(z){}
}par[maxn][maxn][maxn][maxn];
int dp[maxn][maxn][maxn][maxn];
int n, a, b;
int h[maxn]; int main(){
cin >> n >> a >> b;
for (int i = ; i <= n; i++){
scanf("%d", h + i); h[i]++;
}
memset(par, -, sizeof(par));
memset(dp, 0x3f, sizeof(dp));
dp[][h[]][h[]][h[]] = ;
for (int i = ; i <= n - ; i++){
for (int j = h[i - ]; j >= ; j--){
for (int k = h[i]; k >= ; k--){
for (int z = h[i + ]; z >= ; z--){
int nj = max(, j - b), nk = max(, k - a), nz = max(, z - b);
if (dp[i][nj][nk][nz] > dp[i][j][k][z] + ){
dp[i][nj][nk][nz] = dp[i][j][k][z] + ;
par[i][nj][nk][nz] = Point(i, j, k, z);
}
if (nj == && dp[i + ][nk][nz][h[i + ]] > dp[i][j][k][z] + ){
dp[i + ][nk][nz][h[i + ]] = dp[i][j][k][z] + ;
par[i + ][nk][nz][h[i + ]] = Point(i, j, k, z);
}
if (j == && dp[i + ][k][z][h[i + ]] > dp[i][j][k][z]){
dp[i + ][k][z][h[i + ]] = dp[i][j][k][z];
par[i + ][k][z][h[i + ]] = par[i][j][k][z];
}
}
}
}
}
int ans = dp[n][][][];
printf("%d\n", ans);
Point tmp = Point(n, , , );
while (ans--){
tmp = par[tmp.i][tmp.j][tmp.k][tmp.z];
printf("%d ", tmp.i);
}
return ;
}
四维dp 或者 剪枝 + dfs Codeforces Beta Round #6 (Div. 2 Only) D的更多相关文章
- 暴力/DP Codeforces Beta Round #22 (Div. 2 Only) B. Bargaining Table
题目传送门 /* 题意:求最大矩形(全0)的面积 暴力/dp:每对一个0查看它左下的最大矩形面积,更新ans 注意:是字符串,没用空格,好事多磨,WA了多少次才发现:( 详细解释:http://www ...
- Codeforces Beta Round #83 (Div. 1 Only)题解【ABCD】
Codeforces Beta Round #83 (Div. 1 Only) A. Dorm Water Supply 题意 给你一个n点m边的图,保证每个点的入度和出度最多为1 如果这个点入度为0 ...
- Codeforces Beta Round #77 (Div. 2 Only)
Codeforces Beta Round #77 (Div. 2 Only) http://codeforces.com/contest/96 A #include<bits/stdc++.h ...
- Codeforces Beta Round #65 (Div. 2)
Codeforces Beta Round #65 (Div. 2) http://codeforces.com/contest/71 A #include<bits/stdc++.h> ...
- Codeforces Beta Round #59 (Div. 2)
Codeforces Beta Round #59 (Div. 2) http://codeforces.com/contest/63 A #include<bits/stdc++.h> ...
- Codeforces Beta Round #52 (Div. 2)
Codeforces Beta Round #52 (Div. 2) http://codeforces.com/contest/56 A #include<bits/stdc++.h> ...
- Codeforces Beta Round #40 (Div. 2)
Codeforces Beta Round #40 (Div. 2) http://codeforces.com/contest/41 A #include<bits/stdc++.h> ...
- Codeforces Beta Round #22 (Div. 2 Only)
Codeforces Beta Round #22 (Div. 2 Only) http://codeforces.com/contest/22 A 水题 #include<bits/stdc+ ...
- Codeforces Beta Round #14 (Div. 2)
Codeforces Beta Round #14 (Div. 2) http://codeforces.com/contest/14 A 找最大最小的行列值即可 #include<bits/s ...
随机推荐
- 23个mysql查询语句
一查询数值型数据: SELECT * FROM tb_name WHERE sum > 100; 查询谓词:>,=,<,<>,!=,!>,!<,=>,= ...
- js--3目运算符
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name ...
- Spring Security(13)——session管理
1.1 检测session超时 1.2 concurrency-control 1.3 session 固定攻击保护 Spring Security通过http元素下的子元素s ...
- jQuery插件实现左右无缝轮播
前段时间学习jQuery的时候,在网上找了个SuperSlide插件,做轮播图demo,感觉对于新人而言,还是挺容易上手的,代码量也少. 直接贴代码吧. 1.HTML <!DOCTYPE htm ...
- C#字符串转INT
Convent.ToInt32(string ) 可以转化字符串 其他 * .tostring(); (int ) char //强制类型转换 不能转化字符串
- 关于 plsql连接oracle数据库的问题
因电脑没有 安装oracle服务器端,所以安装plsql后还是不能连接oracle数据库,网上下载一个oracle客户端工具,是绿色免安装的,解压就好.我放我云盘了. instantclient_11 ...
- javascript的insertBefore、insertAfter和appendChild简单介绍
target.insertBefore(newChild,existingChild)参数说明:1.target:被添加节点和现有节点的父节点.2.newChild:将要被插入的节点.3.exis ...
- 关于CSS样式的那些事_导航条菜单讲解
最近开始忙着开自己的个人博客了,自己的前端确实是渣渣.没办法,一步步来,从慕课网上慢慢学着先. 首先带来的是一个导航栏的设计: 垂直导航栏的设计: 直接上代码: <!DOCTYPE html P ...
- ERP流程图
- [ An Ac a Day ^_^ ] UVALive 2635 Housing Complexes 二分图最大匹配
快要比赛了 看看原来做过的题 感觉这道题当时做的还是挺费劲的 所以发一下 题意: 一个土豪要建别墅 因为有的地区地方不够大 所以要拆屋子 每个地方的字母就是对应开发商的地盘 没有字母的就是自由土地 一 ...