BZOJ 1634 [Usaco2007 Jan]Protecting the Flowers 护花:贪心【局部分析法】

题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1634

题意:

  约翰留下他的N只奶牛上山采木。可是,当他回来的时候,他看到了一幕惨剧:牛们正躲在他的花园里,啃食着他心爱的美丽花朵!

  为了使接下来花朵的损失最小,约翰赶紧采取行动,把牛们送回牛棚。

  第i只牛所在的位置距离牛棚t[i](1 <= t[i] <= 2000000)分钟的路程,而在约翰开始送她回牛棚之前,她每分钟会啃食e[i](1 <= e[i] <= 100)朵鲜花。

  无论多么努力,约翰一次只能送一只牛回棚。而运送第第i只牛事实上需要2Ti分钟,因为来回都需要时间。

  写一个程序来决定约翰运送奶牛的顺序,使最终被吞食的花朵数量最小。

 

题解:

  贪心。

  对于顺序相邻的两只牛a和b,交换a和b的顺序,对于a和b之外的牛是没有影响的。

  将其他的牛看作一只牛c。

  当a排在b之前时,答案为:

    ans1 = t[a]*(e[b]+e[c]) + t[b]*e[c]

  当b排在a之前时,答案为:

    ans2 = t[b]*(e[a]+e[c]) + t[a]*e[c]

  假设a排在b前面的时候答案更优,则有:

    ans1 < ans2

    即:t[a]*(e[b]+e[c]) + t[b]*e[c] < t[b]*(e[a]+e[c]) + t[a]*e[c]

    整理得:t[a]*e[b] < t[b]*e[a]

  所以按照t[a]*e[b] < t[b]*e[a]排序就好了。

 

AC Code:

 1 // before: t[a]*(e[b]+e[c]) + t[b]*e[c]
 2 // after:  t[b]*(e[a]+e[c]) + t[a]*e[c]
 3 // if a is better:
 4 // t[a]*(e[b]+e[c]) + t[b]*e[c] < t[b]*(e[a]+e[c]) + t[a]*e[c]
 5 // t[a]*e[b] + t[a]*e[c] + t[b]*e[c] < t[b]*e[a] + t[b]*e[c] + t[a]*e[c]
 6 // t[a]*e[b] < t[b]*e[a]
 7 #include <iostream>
 8 #include <stdio.h>
 9 #include <string.h>
10 #include <algorithm>
11 #define MAX_N 100005
12 
13 using namespace std;
14 
15 struct Cow
16 {
17     int eat;
18     int tim;
19     Cow(int _eat,int _tim)
20     {
21         eat=_eat;
22         tim=_tim;
23     }
24     Cow(){}
25     friend bool operator < (const Cow &a,const Cow &b)
26     {
27         return a.tim*b.eat<b.tim*a.eat;
28     }
29 };
30 
31 int n;
32 long long ans=0;
33 Cow cow[MAX_N];
34 
35 void read()
36 {
37     cin>>n;
38     for(int i=0;i<n;i++)
39     {
40         cin>>cow[i].tim>>cow[i].eat;
41         cow[i].tim<<=1;
42     }
43 }
44 
45 void solve()
46 {
47     sort(cow,cow+n);
48     int tot=0;
49     for(int i=0;i<n;i++)
50     {
51         tot+=cow[i].eat;
52     }
53     for(int i=0;i<n;i++)
54     {
55         tot-=cow[i].eat;
56         ans+=cow[i].tim*tot;
57     }
58 }
59 
60 void print()
61 {
62     cout<<ans<<endl;
63 }
64 
65 int main()
66 {
67     read();
68     solve();
69     print();
70 }

 

转载于:https://www.cnblogs.com/Leohh/p/7624502.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值