贪心问题——国王游戏

[NOIP2012 提高组] 国王游戏

题目来源:https://www.luogu.com.cn/problem/P1080

题目描述

恰逢 H 国国庆,国王邀请 n n n 位大臣来玩一个有奖游戏。首先,他让每个大臣在左、右手上面分别写下一个整数,国王自己也在左、右手上各写一个整数。然后,让这 n n n 位大臣排成一排,国王站在队伍的最前面。排好队后,所有的大臣都会获得国王奖赏的若干金币,每位大臣获得的金币数分别是:排在该大臣前面的所有人的左手上的数的乘积除以他自己右手上的数,然后向下取整得到的结果。

国王不希望某一个大臣获得特别多的奖赏,所以他想请你帮他重新安排一下队伍的顺序,使得获得奖赏最多的大臣,所获奖赏尽可能的少。注意,国王的位置始终在队伍的最前面。

输入格式

第一行包含一个整数 n n n,表示大臣的人数。

第二行包含两个整数 a a a b b b,之间用一个空格隔开,分别表示国王左手和右手上的整数。

接下来 n n n 行,每行包含两个整数 a a a b b b,之间用一个空格隔开,分别表示每个大臣左手和右手上的整数。

输出格式

一个整数,表示重新排列后的队伍中获奖赏最多的大臣所获得的金币数。

样例 #1

样例输入 #1

3 
1 1 
2 3 
7 4 
4 6

样例输出 #1

2

提示

【输入输出样例说明】

1 1 1 2 2 2 3 3 3 这样排列队伍,获得奖赏最多的大臣所获得金币数为 2 2 2

1 1 1 3 3 3 2 2 2 这样排列队伍,获得奖赏最多的大臣所获得金币数为 2 2 2

2 2 2 1 1 1 3 3 3 这样排列队伍,获得奖赏最多的大臣所获得金币数为 2 2 2

按$ 2$、 3 3 3、$1 $这样排列队伍,获得奖赏最多的大臣所获得金币数为 9 9 9

3 3 3 1 1 1、$2 $这样排列队伍,获得奖赏最多的大臣所获得金币数为 2 2 2

按$ 3$、 2 2 2 1 1 1 这样排列队伍,获得奖赏最多的大臣所获得金币数为 9 9 9

因此,奖赏最多的大臣最少获得 2 2 2 个金币,答案输出 2 2 2

【数据范围】

对于 20 % 20\% 20% 的数据,有 1 ≤ n ≤ 10 , 0 < a , b < 8 1≤ n≤ 10,0 < a,b < 8 1n10,0<a,b<8

对于 40 % 40\% 40% 的数据,有$ 1≤ n≤20,0 < a,b < 8$;

对于 60 % 60\% 60% 的数据,有 1 ≤ n ≤ 100 1≤ n≤100 1n100

对于 60 % 60\% 60% 的数据,保证答案不超过 1 0 9 10^9 109

对于 100 % 100\% 100% 的数据,有 1 ≤ n ≤ 1 , 000 , 0 < a , b < 10000 1 ≤ n ≤1,000,0 < a,b < 10000 1n1,000,0<a,b<10000

NOIP 2012 提高组 第一天 第二题

思路分析

本道题要求每位大臣的的前面的大臣的左手乘积除当前大臣右手的金币数最小,因此我们可以想到让分子变小/分母变大,所以我们尽量让小的在前,大的在后。总结:对于这样的都希望让二者在前的,我们可以对乘积/和进行排序,当题目中需要乘积的时候,我们就对乘积进行排序,对和进行排序我们可以看 奶牛玩杂技,又因为本道题数据量特别大,因此我们要用高精度来做

代码实现

//由题知:我们可以让除数变大,乘积尽可能的小

#include<iostream>
#include<algorithm>
#include<vector>

#define x first
#define y second

using namespace std;

typedef pair<int,int>PII;

const int N = 1e6+10;

struct E{
    int x,y;
    bool operator<(const E &t)const{
        return x*y<t.x*t.y;
    }
}e[N];

vector<PII> w;
vector<int>res;
vector<int>sum;
int n;
//这道题特殊之处在于高精度

vector<int> mul(vector<int>&a,int b){
    vector<int>c;
    int t=0;
    for(int i=0;i<a.size()||t;i++){
        if(i<a.size())t+=a[i]*b;
        c.push_back(t%10);
        t/=10;
    }
    return c;
}

vector<int> dive(vector<int>&a,int b){
    int t=0;//余数
    vector<int>c;
    for(int i=a.size()-1;i>=0;i--){
        t=t*10+a[i];
        c.push_back(t/b);
        t%=b;
    }
    
    reverse(c.begin(),c.end());
    while(c.size()>1 && c.back()==0)c.pop_back();
    // for(int i=c.size()-1;i>=0;i--)cout<<c[i];
    //     puts("");
    return c;
}

int cmp(vector<int>&a,vector<int>&b){
    if(a.size()>b.size())return 1;
    if(a.size()<b.size())return -1;
    for(int i=0;i<a.size();i++){
        if(a[i]>b[i])return 1;
        if(a[i]<b[i])return -1;
    }
    return 0;//判断相等
}

int main(){
    cin>>n;
    int a,b;
    cin>>a>>b;
    sum.push_back(1);
    sum=mul(sum,a);
    for(int i=1;i<=n;i++){
        int a,b;
        cin>>a>>b;
        e[i]={a,b};
    }
    
    sort(e+1,e+1+n);
    
    for(int i=1;i<=n;i++){
        int fenzi=e[i].x,fenmu=e[i].y;
        // res=max(res,sum/fenmu);
        vector<int>k;
        k=dive(sum,fenmu);
        
        if(cmp(res,k)==-1)res=k;
        sum=mul(sum,fenzi);
        k.clear();
    }
    
    for(int i=res.size()-1;i>=0;i--)cout<<res[i];
    
    return 0;
}
  • 19
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

green qwq

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值