题目地址:https://vjudge.net/problem/HDU-6508

思路:
给一些卡,分为四种卡。
1.白卡(没效果)
2.魔法,作用卡(会对作用卡的费用减少,也会被魔法卡作用)
3.作用卡(会被魔法卡作用使其费用减少)
4.魔法卡(会对作用卡的费用减少)

有一个想法:如果我们得到最大的攻击力,其中会用到魔法卡和作用卡或者两者效果都有的卡的话,魔法卡其实是越早用越好
,而作用卡越晚用晚用越好,因为作用卡会被魔法卡影响,使其费用减少,而且费用少的卡需要先使用,才能尝试更多的可能,
如果一张卡是10w,一张卡是1w,为了尝试更多可能,先让费用少的先尝试,(比如,先1w可以dp[1w],dp[1w+ 10w],但dp[10w],之前没有dp[1w])让费用多的之后对dp[N][N]进行比较。
那么,dp[N][N], 第一个维度为费用值,第二个维度为用了几张魔法卡,维护的是最大攻击力。
然后就是一个01背包问题,当然为了实现我们的想法,需要对结构体进行某些特定的排序。

补:其实可以这么想。我们打出了最好顺序情况的卡,那么我们的打出的卡中有其中几类的卡。那么,如果我们用了魔法卡,那我们一定先用魔法卡,用了作用卡,那么作用卡一定是之后用的,从最好的结果,来反推该背包该怎么dp,之后我看博客,想想这样更好理解,在这里补充一下。


 #include<iostream>
#include<algorithm>
using namespace std;
#define rep(i,j,k) for(int i = (j); i <= (k); i++)
#define per(i,j,k) for(int i = (j); i >= (k); i--) const int N = ;
int dp[N][N]; struct node{
int w, x;
int is_m, is_s; bool friend operator < (node a, node b){
if (a.is_m != b.is_m) return a.is_m > b.is_m;//魔法卡先返回
else if (a.is_s != b.is_s) return a.is_s < b.is_s;//作用卡越晚用越好 return a.w < b.w;//先用费用小的,使其dp可以得到全部可能
}
}; node arr[N]; void init(int x){
// rep(i, 0, x)rep(j, 0, x) dp[i][j] = 0;
memset(dp, , sizeof(dp));
} int main(){ ios::sync_with_stdio(false);
cin.tie(); int n, W;
while (cin >> n >> W){ int m_num = ; rep(i, , n){
cin >> arr[i].w >> arr[i].x >> arr[i].is_m >> arr[i].is_s;
if (arr[i].is_m) ++m_num;//统计魔法卡数量
} init(n); sort(arr + , arr + + n); bool is_s = false;//判断是不是作用卡
//从大到小遍历,可以不影响之前的状态
rep(i, , n){
if (arr[i].is_m){ per(k, i, ){//魔法卡从多到少
per(j, W, ){//费用从多到少 int tmp_w = ;
if (arr[i].is_s){//有被作用效果
is_s = true;
tmp_w = max(, arr[i].w - (k - ));//触发魔法卡效果
}
//对于k张魔法卡,之前只有k-1张魔法卡,假如一张魔法卡去更新dp,所以被作用时
//应该是魔法卡数量-1的费用减少
if (is_s && tmp_w <= j){//是作用卡,费用足够
dp[j][k] = max(dp[j][k],dp[j - tmp_w][k - ] + arr[i].x);
}
else if (!is_s && arr[i].w <= j){//不是作用卡,费用足够
dp[j][k] = max(dp[j][k], dp[j - arr[i].w][k - ] + arr[i].x);
}
}
}
}
else {
//进入该部分时,全部的魔法卡使用情况已经存入dp,之后的都不是魔法卡
per(k, m_num, ){//m_num是魔法卡总数量,可能有不用魔法卡才能的到最大攻击的情况
per(j, W, ){ int tmp_w = ;
if (arr[i].is_s){
is_s = true;
tmp_w = max(, arr[i].w - k);
}
//对于k张魔法卡,被作用卡卡费用减少k
if (is_s && tmp_w <= j){
dp[j][k] = max(dp[j][k], dp[j - tmp_w][k] + arr[i].x);
}
else if (!is_s && arr[i].w <= j){
dp[j][k] = max(dp[j][k], dp[j - arr[i].w][k] + arr[i].x);
}
}
}
} is_s = false;
} int ans = ; rep(i, , m_num) ans = max(ans, dp[W][i]);//遍历费用是W的dp最大值,即为最大攻击力 cout << ans << endl;
} return ;
}

2018年东北地区赛S - Problem I. Spell Boost HDU - 6508的更多相关文章

  1. HDOJ 6508 Problem I. Spell Boost (01背包/DP)

    链接:http://acm.hdu.edu.cn/showproblem.php?pid=6508 题目: Problem Description Shadowverse is a funny car ...

  2. 2018 东北地区大学生程序设计竞赛(ABEHIK)

    HDU6500:Problem A. Game with string 题意: 给你一个字符串s以及它的m个子串的首尾位置,现在Alice和 Bob两个人轮流在任一子串的前面或者后面加1个字符,要求加 ...

  3. HDU 5924 Mr. Frog’s Problem 【模拟】 (2016CCPC东北地区大学生程序设计竞赛)

    Mr. Frog's Problem Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Other ...

  4. 2016 CCPC 东北地区重现赛

    1. 2016 CCPC 东北地区重现赛 2.总结:弱渣,只做出01.03.05水题 08   HDU5929 Basic Data Structure    模拟,双端队列 1.题意:模拟一个栈的操 ...

  5. [2018冬令营模拟测试赛(二十一)]Problem A: Decalcomania

    [2018冬令营模拟测试赛(二十一)]Problem A: Decalcomania 试题描述 输入 见"试题描述" 输出 见"试题描述" 输入示例 见&quo ...

  6. HDU 5923 Prediction(2016 CCPC东北地区大学生程序设计竞赛 Problem B,并查集)

    题目链接  2016 CCPC东北地区大学生程序设计竞赛 B题 题意  给定一个无向图和一棵树,树上的每个结点对应无向图中的一条边,现在给出$q$个询问, 每次选定树中的一个点集,然后真正被选上的是这 ...

  7. HDU 5925 Coconuts 【离散化+BFS】 (2016CCPC东北地区大学生程序设计竞赛)

    Coconuts Time Limit: 9000/4500 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Su ...

  8. HDU 5929 Basic Data Structure 【模拟】 (2016CCPC东北地区大学生程序设计竞赛)

    Basic Data Structure Time Limit: 7000/3500 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Oth ...

  9. HDU 5926 Mr. Frog's Game 【模拟】 (2016CCPC东北地区大学生程序设计竞赛)

    Mr. Frog's Game Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)T ...

随机推荐

  1. 探究操作系统的内存分配(malloc)对齐策略

    问题: 我们在写程序的时候经常发现程序使用的内存往往比我们申请的多,为了优化程序的内存占用,搅尽脑汁想要优化内存占用,可是发现自己的代码也无从优化了,怎么办?现在我们把我们的焦点放到malloc上,毕 ...

  2. Win8 Metro(C#)数字图像处理--3.4图像信息熵计算

    原文:Win8 Metro(C#)数字图像处理--3.4图像信息熵计算 [函数代码] /// <summary> /// Entropy of one image. /// </su ...

  3. 零元学Expression Blend 4 - Chapter 8 用实例了解布局容器系列-「Grid」

    原文:零元学Expression Blend 4 - Chapter 8 用实例了解布局容器系列-「Grid」 本系列将教大家以实做案例认识Blend 4 的布局容器,此章介绍的是Blend 4 里的 ...

  4. UWP-标题栏”后退“按钮

    原文:UWP-标题栏"后退"按钮 标题栏”后退“按钮,系统级导航 应用必须启用所有硬件和软件系统后退按钮的后退导航.执行此操作的方法是注册 BackRequested 事件的侦听器 ...

  5. Android多线程(二)

    在上一篇中,我简单说了用AsyncTask来完成简单异步任务,但AsyncTask是把所有的异步任务放到一个队列中依次在同一个线程中执行.这样就带来一个问题,它无法处理那些耗时长.需要并行的的任务.如 ...

  6. SimpleDateFormat之后为何多了一年,难道Java API也这么不靠谱?

    这一切的背后到底是机器故障,还是程序的bug? 难道Java API也不靠谱 朋友在我博客上发现一时间明显错误,操作时间怎么会是2016年?在同一个for循环输出到页面的时间,唯独这一个时间不对,整整 ...

  7. char、char*、char**数组(有图,非常清楚)good

    平时都用的是char数组,基本忘记了char*数组和char**数组该怎么用了 char s1[10]; s1[0] s1[1]等都是char s1是char*,等同于&s1[0] char* ...

  8. 移动Web - 响应式布局开篇

    用到的工具: FireFox浏览器 Sublime Text 2 响应式布局定义: 2010年,Ethan Marcotte提出,可查看原文: 通俗地讲就是:百份比布局,根据不同设备显示不同布局: 这 ...

  9. Markdown的选择

    直击现场 我一直在思索用什么格式存储文档比较好.之前一直在用google docs,但是它的格式不公开,上传/下载的时候需要转换格式,转换的时候必然会丢失一些信息.于是后来想,那还是纯本文或者mark ...

  10. FMX中实现PostMessage的方法

    首先,做为一个从Windows下转过来的开发人员,许多时候,我们喜欢用PostMessage来触发一些异步执行的代码,但遗憾的是,FMX做为EMB的跨平台库的基础,只提供了SendMessage方法, ...