三分 - HNU 13409 Flowers

本文探讨了在给定N颗初始营养值为0的种子的情况下,通过浇水和施肥来提升其营养值,使得营养值达到特定阈值后开花的优化策略。文章深入分析了此问题的凸函数特性,并提出了利用三分搜索法来确定最优浇水策略,最终通过精确计算得出最佳解决方案。

Flowers 

Problem's Link:  http://acm.hnu.cn/online/?action=problem&type=show&id=13409&courseid=0


 

Mean: 

有N颗种子,每颗种子初始时营养值为0。当一颗种子营养值达到th后就会开花。

有两种操作:

1.给所有的种子浇w升水;

2.给某个种子施f升肥;

对于第i颗种子,每浇1升水会增加vw点营养值,每施1升肥可以增加vf点营养值,该种种子的肥料单价为pf,当营养值达到th后开花。

浇水必须一起浇,而且水的单价是一样的,都是pw。花的th可能为负,而且有的花浇水营养值会减少。

analyse:

很明显的凸函数。

由于浇水的量所有种子都相同,所以三分水量即可。

Time complexity: O(N)

 

Source code: 

/*
* this code is made by crazyacking
* Verdict: Accepted
* Submission Date: 2015-08-16-19.52
* Time: 0MS
* Memory: 137KB
*/
#include <queue>
#include <cstdio>
#include <set>
#include <string>
#include <stack>
#include <cmath>
#include <climits>
#include <map>
#include <cstdlib>
#include <iostream>
#include <vector>
#include <algorithm>
#include <cstring>
#define eps 1e-14
#define  LL __int64
#define  ULL unsigned __int64
using namespace std;
const int MAXN = 10 +( int) 1e5;
LL n , pw , vw [ MAXN ], pf [ MAXN ], vf [ MAXN ], th [ MAXN ];
double tmp , t;
double cal( double & w)
{
      tmp = w * pw;
      for( int i = 0; i <n; ++ i)
      {
            t = th [ i ] - w * vw [ i ];
            if( t <= 0) continue;
            tmp +=( t) * pf [ i ] * 1. / vf [ i ];
      }
      return tmp;
}

double work( double l , double r)
{
      double mid , mmid;
      int st = 100;
      while( st --)
      {
            mid =( l + r) / 2;
            mmid =( mid + r) / 2;
            if( cal( mid) - cal( mmid) > eps)
                  l = mid;
            else r = mmid;
      }
      return l;
}
int main()
{
      ios_base :: sync_with_stdio( false);
      cin . tie( 0);
      while( scanf( "%I64d" , &n ),n)
      {
            scanf( "%I64d" , & pw);
            for( int i = 0; i <n; ++ i)
            {
                  scanf( "%I64d %I64d %I64d %I64d" , & vw [ i ], & pf [ i ], & vf [ i ], & th [ i ]);
            }
            double val = work( 0 , 200);
            printf( "%.6lf \n " , cal( val));
      }
      return 0;
}
/*

*/

 

转载于:https://www.cnblogs.com/crazyacking/p/4735188.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值