关于 耍杂技的牛 一题的思路+代码

题目来源:AcWing 125. 耍杂技的牛&洛谷 P1842 [USACO05NOV] 奶牛玩杂技

题目描述

农民约翰的N头奶牛(编号为1…N)计划逃跑并加入马戏团,为此它们决定练习表演杂技。
奶牛们不是非常有创意,只提出了一个杂技表演:
叠罗汉,表演时,奶牛们站在彼此的身上,形成一个高高的垂直堆叠。
奶牛们正在试图找到自己在这个堆叠中应该所处的位置顺序。
这N头奶牛中的每一头都有着自己的重量 Wi 以及自己的强壮程度 Si
一头牛支撑不住的可能性取决于它头上所有牛的总重量(不包括它自己)减去它的身体强壮程度的值,现在称该数值为风险值,风险值越大,这只牛撑不住的可能性越高。
您的任务是确定奶牛的排序,使得所有奶牛的风险值中的最大值尽可能的小。

输入描述

第一行输入整数N,表示奶牛数量。
接下来N行,每行输入两个整数,表示牛的重量和强壮程度,第 i 行表示第 i 头牛的重量 Wi 以及它的强壮程度 Si

输出描述

输出一个整数,表示最大风险值的最小可能值。

数据范围

1≤ N ≤50000,
1≤ Wi ≤10,000,
1≤ Si ≤1,000,000,000

输入样例

3
10 3
2 5
3 3

输出样例

2

OP

这道题代码不难,主要是思路。

思路

我们可以把排序问题拆分成从输入顺序开始的足够多次对换,只要我们找到能更趋近与所求结果的对换条件,我们便可以依照此条件直接排序。

假设现在有两头牛,i 与 i + 1,我们要找到一种使对换后两牛最大危险值小于对换前的两牛最大危险值的条件。

由题意,对换前两牛( i , i + 1 )危险值分别为
Q i = W 1 + W 2 + . . . + W i − 1 − S i Q_i=W_1+W_2+...+W_{i-1} - S_i Qi=W1+W2+...+Wi1Si

Q i + 1 = W 1 + W 2 + . . . + W i − 1 + W i − S i + 1 Q_{i+1}=W_1+W_2+...+W_{i-1}+W_i - S_{i+1} Qi+1=W1+W2+...+Wi1+WiSi+1

对换后,( i , i + 1 )危险值分别为
H i = W 1 + W 2 + . . . + W i − 1 + W i + 1 − S i H_{i}=W_1+W_2+...+W_{i-1}+W_{i+1} - S_{i} Hi=W1+W2+...+Wi1+Wi+1Si

H i + 1 = W 1 + W 2 + . . . + W i − 1 − S i + 1 H_{i+1}=W_1+W_2+...+W_{i-1} - S_{i+1} Hi+1=W1+W2+...+Wi1Si+1

省略掉相同部分,若要使 M A X ( W i + 1 − S i , − S i + 1 ) ≥ M A X ( − S i , W i − S i + 1 ) MAX(W_{i+1} - S_{i} ,- S_{i+1} )≥MAX( - S_{i} ,W_i - S_{i+1}) MAX(Wi+1Si,Si+1)MAX(Si,WiSi+1)
只需满足 W i + 1 − S i ≥ W i − S i + 1 W_{i+1} - S_{i}≥W_i - S_{i+1} Wi+1SiWiSi+1
移项可得 W i + 1 + S i + 1 ≥ W i + S i W_{i+1}+ S_{i+1}≥W_{i}+ S_{i} Wi+1+Si+1Wi+Si

由此结论,我们可以直接按照 Wi+Si 降序排列,即是最大风险值最小的情况,再遍历求出最大危险值。

代码

#include <bits/stdc++.h>
using namespace std;
struct co{int w,s;}cow[50004];
bool cmp(co a,co b)
{
    return a.w+a.s<b.w+b.s;
}
int main()
{
    int w,s,n,i;
    cin>>n;
    for(i=1;i<=n;i++)
    {
        scanf("%d%d",&cow[i].w,&cow[i].s);
    }
    sort(cow+1,cow+n+1,cmp);
    int mi=0x87777777;//最小值
    long long wi=0;
    for(i=1;i<=n;i++)
    {
        mi=max((long long)mi,wi-cow[i].s);
        wi+=cow[i].w;
    }
    printf("%d",mi);
    return 0;
}

ED

y总yyds

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值