四维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 ...
随机推荐
- 【MySQL】使用 Optimizer Trace 观察SQL执行过程
Optimizer Trace 是MySQL 5.6.3里新加的一个特性,可以把MySQL Optimizer的决策和执行过程输出成文本.输出使用JSON格式,便于程序分析和人类阅读. 使用方法 1) ...
- 一个博客总结的css常见的兼容性问题
http://www.cnblogs.com/jikey/p/4233003.html
- # 欢迎使用Markdown编辑器写博客
似的发射点 甜甜 他inn他 absct{ for i 士大夫似的 胜多负少 import os import sys import subprocess import textwrap if sys ...
- Windows 查看端口占用
查看 Windows 端口占用情况 在 Windows 命令行窗口下执行一下命令 查看所有端口占用情况 netstat -ano 查看特定端口的占用情况 netstat -aon|findstr &q ...
- 【留念贴】Android开发——计算器
[过程] 在电商学霸&&代码女神XuFei的影响下,接触到了关于Android客户端的一些开发,第一次在Android平台搞出了一个App,真的是激动不已,所以必须开个留念贴记录一下. ...
- ios获取相册图片 压缩图片
从摄像头/相册获取图片 刚刚在上面的知识中提到从摄像头/相册获取图片是面向终端用户的,由用户去浏览并选择图片为程序使用.在这里,我们需要过UIImagePickerController类来和用户交互. ...
- iOS中的base64加密
#import <UIKit/UIKit.h> @interface Base64String : NSObject + (NSString *)base64String:(NSStrin ...
- iOS开发瀑布流的实现
//自定义布局,继承于UICollectionViewLayout #import <UIKit/UIKit.h> @class WaterFlowLayout; @protocol W ...
- HDU 2058 The sum problem(枚举)
The sum problem Problem Description Given a sequence 1,2,3,......N, your job is to calculate all the ...
- 在线用户管理--ESFramework 4.0 进阶(05)
无论我们采用何种通信框架来构建我们的分布式系统,在服务端进行用户管理都是非常重要的一个环节.然而用户管理是否应该隶属于通信框架了?这个并不一定,通常来说,用户管理是与具体应用紧密相关的,应该是由应用解 ...