Best Cow Fences

Time Limit: 1000MS Memory Limit: 30000K

Description

Farmer John’s farm consists of a long row of N (1 <= N <= 100,000)fields. Each field contains a certain number of cows, 1 <= ncows <= 2000.

FJ wants to build a fence around a contiguous group of these fields in order to maximize the average number of cows per field within that block. The block must contain at least F (1 <= F <= N) fields, where F given as input.

Calculate the fence placement that maximizes the average, given the constraint.

Input

* Line 1: Two space-separated integers, N and F.

* Lines 2..N+1: Each line contains a single integer, the number of cows in a field. Line 2 gives the number of cows in field 1,line 3 gives the number in field 2, and so on.

Output

* Line 1: A single integer that is 1000 times the maximal average.Do not perform rounding, just print the integer that is 1000*ncows/nfields.

Sample Input

10 6

6

4

2

10

3

8

5

9

4

1

Sample Output

6500

Source

USACO 2003 March Green

/*
01分数规划思想+DP做法.
求长度不小于k连续一段的最大平均值.
ans=(sum[j]-sum[i-1])/(j-i+1) [j-i+1>=k]
这样的话我们二分一个ans,然后让每个数都减去ans.
最后DP检验最大的连续和是否大于0即可.
复杂度O(nlogn).
*/
#include<iostream>
#include<cstdio>
#define MAXN 100001
#define eps 1e-6
using namespace std;
double ans,s[MAXN],sum[MAXN],max1,min1=1e9;
int n,k;
int read()
{
int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9') x=x*10+ch-48,ch=getchar();
return x*f;
}
bool check(double x)
{
double tot,m=0;
for(int i=1;i<=n;i++) sum[i]=sum[i-1]+s[i]-x;
for(int i=k;i<=n;i++)
{
tot=sum[i]-m;
if(tot>=0) return true;
m=min(m,sum[i-k+1]);
}
return false;
}
void erfen(double l,double r)
{
double mid;
while(l+eps<=r)
{
mid=(l+r)/2;
if(check(mid)) ans=mid,l=mid;
else r=mid;
}
int x=r*1000;
printf("%d\n",x);
return ;
}
int main()
{
int x;
while(~scanf("%d %d",&n,&k))
{
max1=0,min1=1e9;
for(int i=1;i<=n;i++)
{
s[i]=read();
sum[i]=sum[i-1]+s[i];
max1=max(max1,s[i]),min1=min(min1,s[i]);
}
erfen(min1,max1);
}
return 0;
}
/*
斜率优化题.
通过此题大概知道了斜率优化是什么.
因为ans=(sum[j]-sum[i-1])/(j-i+1) [j-i+1>=k].
所以我们可以将看做二维平面中的两个点(i-1,sum[i])和(j,sum[j]).
ans即为两点之间的斜率.
我们要让ans最大化.
n^2的暴力可以看做是有一个点t,和点集Gt{x,0<=x<=t-k},
扫描G中的每个点,使斜率最大化.
但是有些点对于决策是没有影响的.
现在有一个定理:
存在三个有序点i,j,k,如果满足k(i,j)>k(k,j)
即j点是直线ik上方的点,那它不会对决策产生影响.
证明的话见2004周源国家集训队论文,讲的很清楚.
这样的话我们维护一个下凸折线就好了.
我们每次都加入位置为i-k的点,维护下凸性.
如果某一点j(0<=j<=i-k)与点i的斜率最大,
则这个点显然一定是下凸折线的切点.
则它之前的点连线斜率较小也不会对决策产生影响
删掉就可以了.
这样的话由于每个点只入队出队一次
所以复杂度是O(n)的.
参考资料 2004周源国家集训队论文.
*/
#include<iostream>
#include<cstdio>
#define MAXN 100001
using namespace std;
double ans,sum[MAXN],max1,min1=1e9;
int n,k;
struct data{double x,y;}q[MAXN];
int read()
{
int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9') x=x*10+ch-48,ch=getchar();
return x*f;
}
double check(data a,data b,data c)
{
return (b.x-a.x)*(c.y-a.y)-(c.x-a.x)*(b.y-a.y);
}
double Get(data a,data b)
{
return (b.y-a.y)/(b.x-a.x);
}
void slove()
{
data x,now;ans=0;
int head=0,tail=0;
for(int i=k;i<=n;i++)
{
x.x=i-k,x.y=sum[i-k];
while(tail-head>=2&&check(q[tail-2],q[tail-1],x)<0) tail--;//维护下凸性.
q[tail++]=x;
now.x=i,now.y=sum[i];
double k=Get(q[head],now);
while(tail-head>0&&Get(q[head+1],now)>k)//删除没用的点.
{
k=Get(q[head+1],now);
head++;
}
ans=max(ans,k);
}
ans*=1000;
int a=ans;
printf("%d\n",a);
}
int main()
{
int x;
while(~scanf("%d %d",&n,&k))
{
for(int i=1;i<=n;i++)
x=read(),sum[i]=sum[i-1]+x;
slove();
}
return 0;
}

Poj 2018 Best Cow Fences(分数规划+DP&&斜率优化)的更多相关文章

  1. POJ 2018 Best Cow Fences (二分答案构造新权值 or 斜率优化)

    $ POJ~2018~Best~Cow~ Fences $(二分答案构造新权值) $ solution: $ 题目大意: 给定正整数数列 $ A $ ,求一个平均数最大的长度不小于 $ L $ 的子段 ...

  2. POJ 2018 Best Cow Fences(二分+最大连续子段和)

    Best Cow Fences Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 14601 Accepted: 4720 Desc ...

  3. BZOJ2402: 陶陶的难题II(树链剖分,0/1分数规划,斜率优化Dp)

    Description Input 第一行包含一个正整数N,表示树中结点的个数.第二行包含N个正实数,第i个数表示xi (1<=xi<=10^5).第三行包含N个正实数,第i个数表示yi ...

  4. POJ 2018 Best Cow Fences(二分答案)

    题目链接:http://poj.org/problem?id=2018 题目给了一些农场,每个农场有一定数量的奶牛,农场依次排列,问选择至少连续排列F个农场的序列,使这些农场的奶牛平均数量最大,求最大 ...

  5. POJ 2018 Best Cow Fences(二分最大区间平均数)题解

    题意:给出长度>=f的最大连续区间平均数 思路:二分这个平均数,然后O(n)判断是否可行,再调整l,r.判断方法是,先求出每个数对这个平均数的贡献,再求出长度>=f的最大贡献的区间,如果这 ...

  6. POJ 2018 Best Cow Fences

    斜率优化. 设$s[i]$表示前缀和,$avg(i,j)=(s[j]-s[i-1])/(j-(i-1))$.就是$(j,s[j])$与$(i-1,s[i-1])$两点之间的斜率. 如果,我们目前在计算 ...

  7. 【BZOJ-4518】征途 DP + 斜率优化

    4518: [Sdoi2016]征途 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 230  Solved: 156[Submit][Status][ ...

  8. 【BZOJ-3437】小P的牧场 DP + 斜率优化

    3437: 小P的牧场 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 705  Solved: 404[Submit][Status][Discuss ...

  9. 【BZOJ-1010】玩具装箱toy DP + 斜率优化

    1010: [HNOI2008]玩具装箱toy Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 8432  Solved: 3338[Submit][St ...

随机推荐

  1. 3.03定义常量之enum

    [注:本程序验证是使用vs2013版] #include <stdio.h> #include <stdlib.h> #include <string.h> #pr ...

  2. python模块之openpyxl

    这是一个第三方库,可以处理xlsx格式的Excel文件.pip install openpyxl安装.如果使用Aanconda,应该自带了. 读取Excel文件 需要导入相关函数. from open ...

  3. js的for循环中出现异步函数,回调引用的循环值始终是最后的值

    一.问题 今天工作中解决bug发现是由“for循环的异步函数,回调引用的循环值始终是最后的值”的现象导致的,如: for (var i = 0; i < files.length; i++) { ...

  4. zookeeper集群搭建及ZAB协议

    zookeeper集群搭建非常简单,准备三台安装好zookeeper服务器,在其zoo.cfg配置中分表添加如下配置 initLimit 10 集群中的follower与leader之间完成初始化同步 ...

  5. python超链接抓取工具

    python实现自动抓取某站点内所有超链接 (仅供学习使用) 代码部分 #!/usr/bin/python import requests import time import re import s ...

  6. UCOSIII优先级反转

    反转现象 任务优先级:H>M>L 绿色部分:任务占用共享资源 理想状态:7释放信号量后,最高优先级H任务抢占CPU 反转原因:H和L等待同一个信号量,H的任务优先级被降至和L相同优先级,此 ...

  7. java web编程 servlet3

     

  8. Linux下环境变量设置 (转)

    Linux下环境变量设置 1.在Windows 系统下,很多软件安装都需要配置环境变量,比如 安装 jdk ,如果不配置环境变量,在非软件安装的目录下运行javac 命令,将会报告找不到文件,类似的错 ...

  9. c# 计算目录的大小

  10. c# CryptoStream 类