2024秋招刷题记录-可以后悔的贪心

可以后悔的贪心

链接:https://ac.nowcoder.com/acm/problem/50439
来源:牛客网

题目描述
在一个游戏中,tokitsukaze需要在n个士兵中选出一些士兵组成一个团去打副本。
第i个士兵的战力为v[i],团的战力是团内所有士兵的战力之和。
但是这些士兵有特殊的要求:如果选了第i个士兵,这个士兵希望团的人数不超过s[i]。(如果不选第i个士兵,就没有这个限制。)
tokitsukaze想知道,团的战力最大为多少。
输入描述:

第一行包含一个正整数n(1≤n≤10^5)。
接下来n行,每行包括2个正整数v,s(1≤v≤10^9,1≤s≤n)。

输出描述:

输出一个正整数,表示团的最大战力。

示例1
输入
复制

2
1 2
2 2

输出
复制

3

示例2
输入
复制

3
1 3
2 3
100 1

输出
100
现在有了一点点经验,堆和贪心通常相关,如果见到这种贪心问题,求最大值,堆和dp能解决大多数(废话说到这里了)
这里刚开始看到这道题完全没有思路,也没有将这道题与堆联系起来,导致尝试了几次都无法解决,还是很菜的看了题解
这道题的解法出乎意料的简单,即把队伍容量从大到小排,如果有一个超过了,就把价值最小的踢出去,然后记录最大值即可,经过之前的锻炼 实现起来也就不困难了,这里附上我的代码(之后的的此类题也会在这里更新owo)

#include<bits/stdc++.h>
using namespace std;
#define ll long long
priority_queue<ll,vector<ll>, greater<ll> > q;
vector<ll> ve[100005];
int main()
{
    int n;
    cin>>n;
    ll v,s;
    for(int i=0;i<n;i++)scanf("%lld%lld",&v,&s),ve[s].push_back(v);
    ll ans=0,sum=0;
    for(int i=n;i>=1;i--)
    {
         for(auto a:ve[i]){sum+=a;q.push(a);}
         cout<<sum<<endl;
         while (q.size()>i)
         {
            /* code */
            sum-=q.top();q.pop();

         }
         ans=max(ans,sum);
    }
    printf("%lld",ans);

}

结束

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C语言多机调度问题是指将n个作业分配给m台机器,使得完成所有作业的时间最短。这是一个经典的贪心算法问题。在这个问题中,我们需要将n个作业分配给m台机器,使得每台机器的工作时间尽可能平均,从而使得完成所有作业的时间最短。具体实现可以使用两种方法:机器数大于待分配作业数和机器数小于待分配作业数。前者可以直接将每个作业分配给一台机器,后者则需要使用贪心算法,将每个作业分配给当前工作时间最短的机器。 代码实现如下(C语言): ``` #include <stdio.h> #define N 8 //作业数 #define M 5 //机器数 int s[M] = {0,0,0};//每台机器当前已分配的作业总耗时 int main(void) { int time[N] = {17,15,11,9,8,5,4,2};//处理时间按从大到小排序 int maxtime = 0; if(M >= N) { maxtime = setwork1(time,N); } else { maxtime = setwork2(time,N); } printf("最多耗费时间%d。",maxtime); } //机器数大于待分配作业数 int setwork1(int t[],int n) { int i; for(i=0;i<n;i++) { s[i] = t[i]; } int ma = max(s,N); return ma; } //机器数小于待分配作业数 int setwork2(int t[],int n) { int i; int mi = 0; for(i=0;i<n;i++) { mi = min(M); printf("第%d号作业,时间和最小的机器号为%d.时间和为%d:\n",i,mi,s[mi]); s[mi] = s[mi]+t[i]; } int ma = max(s,M); return ma; } //求出目前处理作业的时间和最小的机器号 int min(int m) { int min = 0; int i; for(i=1;i<m;i++) { if(s[min] > s[i]) { min = i; } } return min; } //求最终结果(最长处理时间) int max(int s[],int num) { int max = s[0]; int i; for(i=1;i<num;i++) { if(max < s[i]) { max = s[i]; } } return max; } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值