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的更多相关文章

  1. 暴力/DP Codeforces Beta Round #22 (Div. 2 Only) B. Bargaining Table

    题目传送门 /* 题意:求最大矩形(全0)的面积 暴力/dp:每对一个0查看它左下的最大矩形面积,更新ans 注意:是字符串,没用空格,好事多磨,WA了多少次才发现:( 详细解释:http://www ...

  2. Codeforces Beta Round #83 (Div. 1 Only)题解【ABCD】

    Codeforces Beta Round #83 (Div. 1 Only) A. Dorm Water Supply 题意 给你一个n点m边的图,保证每个点的入度和出度最多为1 如果这个点入度为0 ...

  3. Codeforces Beta Round #77 (Div. 2 Only)

    Codeforces Beta Round #77 (Div. 2 Only) http://codeforces.com/contest/96 A #include<bits/stdc++.h ...

  4. Codeforces Beta Round #65 (Div. 2)

    Codeforces Beta Round #65 (Div. 2) http://codeforces.com/contest/71 A #include<bits/stdc++.h> ...

  5. Codeforces Beta Round #59 (Div. 2)

    Codeforces Beta Round #59 (Div. 2) http://codeforces.com/contest/63 A #include<bits/stdc++.h> ...

  6. Codeforces Beta Round #52 (Div. 2)

    Codeforces Beta Round #52 (Div. 2) http://codeforces.com/contest/56 A #include<bits/stdc++.h> ...

  7. Codeforces Beta Round #40 (Div. 2)

    Codeforces Beta Round #40 (Div. 2) http://codeforces.com/contest/41 A #include<bits/stdc++.h> ...

  8. Codeforces Beta Round #22 (Div. 2 Only)

    Codeforces Beta Round #22 (Div. 2 Only) http://codeforces.com/contest/22 A 水题 #include<bits/stdc+ ...

  9. Codeforces Beta Round #14 (Div. 2)

    Codeforces Beta Round #14 (Div. 2) http://codeforces.com/contest/14 A 找最大最小的行列值即可 #include<bits/s ...

随机推荐

  1. Http错误代码表

    状态码 描述 100 Continue初始的请求已经接受,客户应当继续发送请求的其余部分 101 Switching Protocols服务器将遵从客户的请求转换到另外一种协议 200 OK指示请求成 ...

  2. Recover the String

    Recover the String 题目链接:http://codeforces.com/contest/709/problem/D 构造 这题乍一看很难构造,但是如果知道了整个字符串中'0'和'1 ...

  3. R绘图学习笔记(二)-

    依照计量对比药物A和药物B的响应情况 #计量向量 dose <- c(20,30,40,45,60) #药物A的反应向量数据 drugA <- c(16,20,27,40,60) #药物B ...

  4. KMP详解之二

    KMP算法详解 如果机房马上要关门了,或者你急着要和MM约会,请直接跳到第六个自然段. 我们这里说的KMP不是拿来放电影的(虽然我很喜欢这个软件),而是一种算法.KMP算法是拿来处理字符串匹配的.换句 ...

  5. 2016年团体程序设计天梯赛-决赛 L1-1. 正整数A+B(15)

    本题的目标很简单,就是求两个正整数A和B的和,其中A和B都在区间[1,1000].稍微有点麻烦的是,输入并不保证是两个正整数. 输入格式: 输入在一行给出A和B,其间以空格分开.问题是A和B不一定是满 ...

  6. hdu_2825_Wireless Password(AC自动机+状压DP)

    题目链接:hdu_2825_Wireless Password 题意: 给你m个串,问长度为n至少含k个串的字符串有多少个 题解: 设dp[i][j][k]表示考虑到长度为i,第j个自动机的节点,含有 ...

  7. lua 中 IO库

    read函数从当前输入文件读取串,由它的参数控制读取的内容: "*all”             读取整个文件 "*line"            读取下一行 &qu ...

  8. ocean所用的蝴蝶纹理

    #include <ork/render/FrameBuffer.h> #include <ork/scenegraph/SceneManager.h> #include &l ...

  9. LeetCode OJ 41. First Missing Positive

    Given an unsorted integer array, find the first missing positive integer. For example,Given [1,2,0]  ...

  10. 正则表达式 Pattern & Matcher

    1 compile and pattern Pattern类用于创建一个正则表达式,也可以说创建一个匹配模式,它的构造方法是私有的,不可以直接创建,但可以通过Pattern.complie(Strin ...