xinjun与阴阳师

xinjun与阴阳师

https://ac.nowcoder.com/acm/problem/14602

来源:牛客网

@[toc]

题目描述

xinjun是各类手游的狂热粉丝,因随手一氪、一氪上千而威震工大,现在他迷上了阴阳师。xinjun玩手游有一个习惯,就是经过层层计算制定出一套方案来使操作利益最大化(因此即使有扫荡券也不用,故称圣雄肝帝)。已知阴阳师有N个模式可以操作,模式i有ai种操作,但每种模式每日只能选用一种操作,可以不选。操作j能收益vj,但需要花费体力wj点。xinjun每日拥有体力M点,求他每日最多能得到多少收益。

输入描述:

第一行一个正整数T(T<=10),表示共有T组数据。

对于每组数据,第一行两个正整数N,M(1<=N,M<=1000)。

接下来N段数据,每段第一行一个正整数ai(1<=ai<=1000),第二行ai个正整数vj(1<=vj<=1000),第三行ai个正整数wj(1<=wj<=1000)。

每组数据ai之和不大于104。

输出描述:

对每组数据输出一行,即xinjun每日最多能得到多少收益。

示例1
输入

1
3 10
2
2 3
3 2
2
1 1
3 4
1
5
5

输出

9

题解:

01背包,但是独特点在于,N个模式,每个模式有ai种操作,一个模式只能用一次操作,也就是并非所有操作都参与最终受益,相同模式下只能选一个操作。
这怎么解决呢?
我们用二维数字来存受益与花费
v[i][j]表示第i个模式下第j种操作的受益
w[i][j]表示第i个模式下第j种操作的花费
那么递推方程就是
f[j] = max(f[j],f[j-w[i][k]]+v[i][k]);
三重for循环的含义
在当前容量下,枚举每一种操作,保留最佳情况
我原本以为复杂度会超,发现我想太多了,数据有点水。。
这种方法应该是算是比较简单明了

代码:

#include<bits/stdc++.h>

using namespace std;

const int maxn = 1005;
int n,m;
int f[maxn];
int c[maxn];
int w[maxn][maxn];
int v[maxn][maxn];
int t;

int main()
{
    cin >> t;
    while(t--){
        cin >> n >> m;
        for(int i = 1; i <= n; ++i)
        {
            cin >> c[i];
            for(int j = 1; j <= c[i]; ++j)
            {
                cin >> v[i][j];
            }
            for(int j = 1; j <= c[i]; ++j)
            {
                cin >> w[i][j];
            }
        }
        memset(f,0,sizeof(f));
        f[0] = 0;
        for(int i = 1; i <= n; ++i)//模式 
        {
            for(int j = m; j >= 0; --j)
            {
                for(int k = 1; k <= c[i]; ++k)//第i种操作 
                {
                    if(j>=w[i][k])
                        f[j] = max(f[j],f[j-w[i][k]]+v[i][k]);
                }
            }
        }

        cout << f[m] << endl;
    }
    return 0;
}

还有个方式:
参考
模式与模式之间都是独立的
每一个模式不是只能选一种吗?那我就挑选出最佳的操作,然后进入下一个模式,一直这样走

#include<iostream>
#include<cstring>
using namespace std;
const int N = 1010;
struct node{
    int f;
    int ne;
}dp[N];

int main (){
    int T;
    cin>>T;
    while(T--){
        int P,W;
        scanf("%d %d",&P,&W);
        while(P--){
            int n;
            scanf("%d",&n);
            int w[n],v[n];
            for(int i=0;i<n;i++){
                scanf("%d",&v[i]);
            }
            for(int i=0;i<n;i++){
                scanf("%d",&w[i]);
            }

            for(int i=0;i<n;i++){
                for(int j=W;j>=w[i];j--){
                    dp[j].ne=max(dp[j].ne,dp[j-w[i]].f+v[i]);

                }
            }
            for(int i=0;i<=W;i++){
                dp[i].f=dp[i].ne;
            }
        }

        cout<<dp[W].f<<endl;
        memset(dp,0,sizeof dp);
    }
    return 0;
}
全部评论
为什么for(j=0;j<=m;j++)是错的?有点想不明白,求大佬解答
点赞 回复 分享
发布于 2022-04-25 20:34
这个怎么确定的不会选择同一种模式下不同的操作呀,没想明白,求大佬解答
点赞 回复 分享
发布于 2020-10-02 10:46

相关推荐

避坑恶心到我了大家好,今天我想跟大家聊聊我在成都千子成智能科技有限公司(以下简称千子成)的求职经历,希望能给大家一些参考。千子成的母公司是“同创主悦”,主要经营各种产品,比如菜刀、POS机、电话卡等等。听起来是不是有点像地推销售公司?没错,就是那种类型的公司。我当时刚毕业,急需一份临时工作,所以在BOSS上看到了千子成的招聘信息。他们承诺无责底薪5000元,还包住宿,这吸引了我。面试的时候,HR也说了同样的话,感觉挺靠谱的。于是,我满怀期待地等待结果。结果出来后,我通过了面试,第二天就收到了试岗通知。试岗的内容就是地推销售,公司划定一个区域,然后你就得见人就问,问店铺、问路人,一直问到他们有意向为止。如果他们有兴趣,你就得摇同事帮忙推动,促进成交。说说一天的工作安排吧。工作时间是从早上8:30到晚上18:30。早上7点有人叫你起床,收拾后去公司,然后唱歌跳舞(销售公司都这样),7:55早课(类似宣誓),8:05同事间联系销售话术,8:15分享销售技巧,8:30经理训话。9:20左右从公司下市场,公交、地铁、自行车自费。到了市场大概10点左右,开始地推工作。中午吃饭时间大约是12:00,公司附近的路边盖饭面馆店自费AA,吃饭时间大约40分钟左右。吃完饭后继续地推工作,没有所谓的固定中午午休时间。下午6点下班后返回公司,不能直接下班,需要与同事交流话术,经理讲话洗脑。正常情况下9点下班。整个上班的一天中,早上到公司就是站着的,到晚上下班前都是站着。每天步数2万步以上。公司员工没有自己的工位,百来号人挤在一个20平方米的空间里听经理洗脑。白天就在市场上奔波,公司的投入成本几乎只有租金和工资,没有中央空调。早上2小时,晚上加班2小时,纯蒸桑拿。没有任何福利,节假日也没有3倍工资之类的。偶尔会有冲的酸梅汤和西瓜什么的。公司的晋升路径也很有意思:新人—组长—领队—主管—副经理—经理。要求是业绩和团队人数,类似传销模式,把人留下来。新人不能加微信、不能吐槽公司、不能有负面情绪、不能谈恋爱、不能说累。在公司没有任何坐的地方,不能依墙而坐。早上吃早饭在公司外面的安全通道,未到上班时间还会让你吃快些不能磨蹭。总之就是想榨干你。复试的时候,带你的师傅会给你营造一个钱多事少离家近的工作氛围,吹嘘工资有多高、还能吹自己毕业于好大学。然后让你早点来公司、无偿加班、抓住你可能不会走的心思进一步压榨你。总之,大家在找工作的时候一定要擦亮眼睛,避免踩坑!———来自网友
qq乃乃好喝到咩噗茶:不要做没有专业门槛的工作
点赞 评论 收藏
分享
评论
4
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务