https://www.luogu.org/problem/show?pid=1080

题目描述

恰逢 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≤100;

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

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

NOIP 2012 提高组 第一天 第二题

贪心

设 i, j=i+1     tot为i之前的所有人的左手积   Li,Ri Lj Rj

则 j 的奖赏为 tot*Li/Rj     若交换 i与j     i  的奖赏为 tot*Lj/Ri

尽量使前面的大臣奖赏少的话    应是   tot*Li/Rj < tot*Lj/Ri    即 Li*Ri<Lj*Rj

 #include <algorithm>
#include <cstdio> using namespace std; const int N();
int n,ans,tot=; struct Node
{
int l,r;
}p[N];
bool cmp(Node a,Node b)
{
return a.l*a.r<b.l*b.r;
} int main()
{
scanf("%d",&n);
scanf("%d%d",&p[].l,&p[].r);
for(int i=;i<=n;i++) scanf("%d%d",&p[i].l,&p[i].r);
sort(p+,p+n+,cmp);
for(int i=;i<n;i++)
{
tot=(long long)tot*p[i].l;
ans=max(ans,tot/p[i+].r);
}
printf("%lld",ans);
return ;
}

60

高精AC、、、数组开大、、e xin

 #include <algorithm>
#include <cstring>
#include <cstdio> using namespace std; const int N();
int n; struct Node
{
int l,r;
}p[N];
bool cmp(Node a,Node b)
{
return a.l*a.r<b.l*b.r;
} struct Gj
{
int num[];
void init()
{
memset(num,,sizeof(num));
num[]=; num[]=;
}
void mul(int x)
{
for(int i=,ove=;i<=num[];i++)
{
num[i]=num[i]*x+ove;
if(num[i]>)
{
ove=num[i]/;
num[i]%=;
num[]=max(num[],i+);
}
else ove=;
}
for(;!num[num[]];) num[]--;
}
void del(int x)
{
for(int t=,i=num[];i;i--)
{
int tt=num[i];
num[i]=(t*+tt)/x;
t=(t*+tt)%x;
}
for(;!num[num[]];) num[]--;
}
void print()
{
for(int i=num[];i;i--) printf("%d",num[i]);
} };
Gj temp,ans;
Gj MAX(Gj a,Gj b)
{
if(a.num[]>b.num[]) return a;
else if(a.num[]<b.num[]) return b;
for(int i=a.num[];i;i--)
{
if(a.num[i]>b.num[i]) return a;
else if(a.num[i]<b.num[i]) return b;
}
return a;
} int main()
{
scanf("%d",&n);
scanf("%d%d",&p[].l,&p[].r);
for(int i=;i<=n;i++) scanf("%d%d",&p[i].l,&p[i].r);
sort(p+,p+n+,cmp);
temp.init(); ans.num[]=;
for(int i=;i<n;i++)
{
temp.mul(p[i].l);
Gj tmp=temp; tmp.del(p[i+].r);
ans=MAX(ans,tmp);
}
ans.print();
return ;
}
05-11 21:45