贪心算法——国王游戏

  • 1、题目描述
  • 2、问题分析
  • 3、算法源码

1、题目描述

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

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

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

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

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

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

输入输出样例
输入 #1 复制
3
1 1
2 3
7 4
4 6
输出 #1 复制
2
说明/提示
【输入输出样例说明】

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

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

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

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

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

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

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

【数据范围】

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

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

对于 60%的数据,有 1≤ n≤10;

对于 60%的数据,保证答案不超过 10^9;

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

NOIP 2012 提高组 第一天 第二题

2、问题分析

先利用数学思想进行建模,把问题的一般化形式表示出来,再根据一般化的形式进行分析,从而解析整体化的问题。

设第i个大臣的收益为ci,左手金额为ai,右手金额为bi,第j个大臣的收益为cj,左手金额外aj,右手金额为bj,第i个大臣前i-1个人(包括第1个固定的国王和前i-2个大臣)的金额之和为x。

                      左手金额       右手金额       前i-1个人              x            /第i个大臣              ai           bi第j个大臣              aj           bj

根据题目的要求,有ci = x/bi, cj = x*ai/bj

此时两位大臣的中的较大收益记为C1, 则 C1 = max{ x/bi, x*ai/bj}

因为我们是要研究排列的方式对整体最大收益的影响,因而改变第i个大臣和第j个大臣的排列

                      左手金额       右手金额       前i-1个人              x            /第j个大臣              aj           bj第i个大臣              ai           bi

根据题目的要求,有ci = x*aj/bi,cj = x/bj

此时两位大臣的中的较大收益记为C2,则C2=max{x*aj/bi, x/bj}

(1)、设C1为C1,C2中的较小者,则有C1<=C2,

即 max{x/bi, x* ai/bj} <= max{x*aj/bi,x/bj}

因为x,ai,bi,aj,bj均为大于0的正整数,

x/bi <= x* aj/ bi , x*ai/bj>=x/bj

设k1 = x/bi,k2 = x* ai/bj,k3 = x* aj/bi,k4 = x/bj

则k1<=k3,k2>=k4

为使得max{k1,k2}<=max{k3,k4},则有k2<=k3

即x* ai/bj<=x*aj/bi,整理得ai * bi<=aj * bj

(2)、设C2为C1,C2中的较小者,则有C1>=C2,

即 max{x/bi, x* ai/bj} >= max{x*aj/bi,x/bj}

因为x,ai,bi,aj,bj均为大于0的正整数,

x/bi <= x* aj/ bi , x*ai/bj>=x/bj

设k1 = x/bi,k2 = x* ai/bj,k3 = x* aj/bi,k4 = x/bj

则k1<=k3,k2>=k4

为使得max{k1,k2}>=max{k3,k4},则有k3<=k2

即x* aj/bi<=x*ai/bj,整理得aj * bj<=ai * bi

综上所述,为使大臣中获得较大收益者的收益较小,则对于任意相邻的两个大臣i和j,均满足关系式ai * bi<=aj * bj,即按左右手金额数目乘积排序即可。

(但进行乘法运算的时候请注意,若乘积过程中数值超过了正常数据的表示范围,请采用高精度算法编写程序)

3、算法源码

#include<bits/stdc++.h>
#define ll long long
using namespace std;
int n;
int len = 1;//表示高精度数据的长度
int sum[100001] = {0,1};//高精度数据计算结果
struct minister{ll left;ll right;
} m[1000001];
bool cmp(minister a,minister b){return a.left * a.right < b.left * b.right;
}
void multiplicative(ll x){//高精度乘法 for(int i = 1;i <= len; i++){sum[i] *= x; //各位数乘x,如1111×20 = (1000+100+10+1)×20 } for(int i = 1;i <= len; i++){sum[i + 1] += sum[i] / 10;//获得进位,因为相乘结果可能为 10 3 13 4,需转化为 1 0 4 3 4sum[i] %= 10;}len++;//因为是sum[i+1],所以肯定会计算到sum[len+1],所以需要len++while(sum[len] / 10){//观察最高位是否有进位,因为可能有经过前面的操作后可能有 100 3的情况  sum[len + 1] = sum[len] / 10;sum[len] %= 10;len++; } //经过前面的一波操作后,最后可能会导致sum[len]的位置上有0,若为0则需要消除if(sum[len] == 0)len--;
}
void division(){//高精度除法,这里只需要进行一次操作,所以不需要参数 //思路依旧很简单,如2345/5=(2000+300+40+5)/5,小学生除法 //->2 3 4 5 ->0 23(3 + 2%5*10) 4 5 -> 0 4 34(4 + 23%5*10) 5 -> 0 4 6 45(5 + 34%5*10)for(int i = len; i >= 1; i--){sum[i - 1] += (sum[i] % m[n].right * 10);sum[i] /= m[n].right;}//这波操作必然导致一堆前导0,需消除 while(!sum[len]){len--;}//防止除完了if(len == 0) cout << "1" << endl;
}
int main(){cin >> n;cin >> m[0].left >> m[0].right;for(int i = 1;i <= n; i++)cin >> m[i].left >> m[i].right;sort(m + 1, m + 1 + n, cmp);for(int i = 0;i < n; i++){multiplicative(m[i].left); }division();for(int i = len; i >= 1; i--)cout << sum[i];}

贪心算法——国王游戏(洛谷P1080)相关推荐

  1. 贪心算法——国王游戏

    题目描述 孙悟空给花果山的小猴子们分桃子. 首先,他让每只小猴在左.右手上面分别写下一个整数,悟空自己也在左.右手上各写一个整数. 然后,让这 n 只小猴排成一排,悟空站在队伍的最前面. 排好队后,所 ...

  2. 贪心算法——部分背包(洛谷 P2240)

    贪心算法--部分背包问题 部分背包问题,顾名思义,部分 就是可以取一部分,也就是可以随意拆分的物品,是最简单经典的贪心问题. 解题步骤: 1)用结构体数组保存每个物品的价值及总重量.平均价值: 2)输 ...

  3. 洛谷P1080 国王游戏(贪心)

    国王游戏 题目描述 恰逢 HHH 国国庆,国王邀请 nnn 位大臣来玩一个有奖游戏.首先,他让每个大臣在左.右手上面分别写下一个整数,国王自己也在左.右手上各写一个整数.然后,让这 nnn 位大臣排成 ...

  4. 洛谷 P1080 国王游戏

    这是一道贪心题,贪心的策略是将大臣们按左右手金币的乘积升序排列,具体证明过程可以参见洛谷大佬的题解,这里就不再赘述了. 因为本菜鸡之前没有接触过高精度运算,对C++的运算符重载也不太熟练,所以正好借此 ...

  5. 深入理解 操作系统 SJF算法(以洛谷P1223题为例)

    CPU Scheduling Algorithms 重要的CPU调度算法如下: FCFS Scheduling(First-Come, First-Served) SJF Scheduling(Sho ...

  6. 模拟——扫雷游戏(洛谷 P2670)

    模拟算法指的是让程序完整地按照题目叙述的方式运行得到答案! 此题选自洛谷P2670 也是一道简单模拟的题,没有必要列举出8个if语句逐一判断8个方向, 只需要用一个二维数组来保存8个方向即可. 需要注 ...

  7. 【贪心】国王游戏(ybtoj 贪心-1-4)

    国王游戏 ybtoj 贪心-1-4 题目大意 有一个国王和n个大臣 每人左右手分别有一个数,现在然你对大臣们排列(国王在第一个) 每个大臣所得金币是前面的人左手上的数的积除以他右手上的数 现在问你获得 ...

  8. 深入理解 操作系统 LRU算法(以洛谷P1540题为例)

    LRU算法 LeastRecentlyUsedLeast Recently UsedLeastRecentlyUsed 算法,意为"最近最少使用",这是操作系统内存管理部分重要的一 ...

  9. 广度优先搜索——好奇怪的游戏(洛谷 P1747)

    题目选自洛谷P1747 简单的广搜模板题,4+8 = 12个方向进行bfs,目的地是(1,1) 每次查看队首是否到达,若到达(1,1) 则返回队首步长即可~ 需要注意的是,马走日和像走田的位置计算 i ...

  10. (快速幂算法+高精度)洛谷P1045 麦森数

    前言   故事的最后,让我们以一道十分经典的题目--<麦森数>来结尾.接受现实吧,总会有我们没准备过的高精度运算出现.我们固然可以提前把高精度的快速幂模板也准备好,但是总会有百密一疏的时候 ...

最新文章

  1. sys.check_constraints
  2. Redis 缓存设计原则
  3. objectdatasource中delete的尴尬。
  4. 在C#中,如何将一种编码的字符串转换成另外一种编码。
  5. TCP/IP详解--学习笔记(13)-TCP坚持定时器,TCP保活定时器
  6. Java读书笔记05 类与对象
  7. java多线程启动BIO和NIO服务端同时启动接受图片和字符数据
  8. 克服VR眩晕之帧数:提升UE4内容实时渲染效率
  9. OpenGL基础27:网格
  10. Ubuntu20.04安装ros教程(实测有用)
  11. Java实现微信扫一扫
  12. 软件开发常用英文词汇
  13. Scala折叠(fold)
  14. 现代软件工程-构建之法---第一章 练习与讨论
  15. GNSS 使用DFT算法 能量损耗仿真
  16. java计算机毕业设计Internet快递柜管理系统源码+mysql数据库+系统+lw文档+部署
  17. “微盟事件”星瑞格(Sino-DB)产品解决方案
  18. mendelay为什么安装不了_方舟生存进化手游国际版出现解析包错误和解析包错误以及下载完无法安装这样应对...
  19. nvme协议 sata接口_nvme和sata协议游戏测评
  20. javax.el.PropertyNotFoundException: Property 'eid' not found on type com.aqd.entity.User

热门文章

  1. 关于Chrome浏览器主页被2345篡改
  2. Wampserver修改默认浏览器、默认编辑器的解决办法
  3. Uncaught TypeError: $(...).modal is not a function
  4. matlab rlc串联,rlc串联电路的零状态响应 matlab
  5. 饥荒:进阶·放火烧树
  6. AtCoder Beginner Contest 158 D.String Formation
  7. windows7系统的时间服务器,win7系统搭建ntp服务器的操作方法
  8. 深入浅出理解SVM支持向量机
  9. linux 字符 拨号上网,LINUX下用ADSL拨号上网
  10. 2020 阿里、字节iOS面试题之Runtime相关问题2