zoukankan      html  css  js  c++  java
  • 【BZOJ3316】JC loves Mkk 分数规划+单调队列

    【BZOJ3316】JC loves Mkk

    Description

    Input

    第1行,包含三个整数。n,L,R。
    第2行n个数,代表a[1..n]。

    Output

    仅1行,表示询问答案。
    如果答案是整数,就输出整数;否则,输出既约分数“P/Q”来表示。

    Sample Input

    5 3 4
    3 1 2 4 5

    Sample Output

    7/2
    HINT
    1≤L≤R≤n≤10^5,0≤ai≤10^9,保证问题有解,数据随机生成

    题解:直接二分答案,然后每个糖果的权值都变成a[i]-mid,我们需要找到一段长度在[L,R]中的区间使得权值和>=0。然后我们将区间和转变成前缀相减的形式,所以只需要找到s[j]<s[i],j<i这样的i,j就行了。那么对于每个s[i],我们肯定是贪心地选取前面最小的s[j],这个用单调队列维护即可。

    但是要求区间长度是偶数,所以我们需要开对奇偶各开一个单调队列。同时要求答案是分数,这个是需要再最后算一下就行了。

     

    #include <cstdio>
    #include <cstring>
    #include <iostream>
    typedef long long ll;
    using namespace std;
    const int maxn=100010;
    int n,L,R,h1,t1,h2,t2;
    ll ans1,ans2,g;
    ll A[maxn<<1],S[maxn<<1];
    double v[maxn<<1],s[maxn<<1];
    int q1[maxn<<1],q2[maxn<<1];
    ll gcd(ll a,ll b)
    {
    	return (!b)?a:gcd(b,a%b);
    }
    bool check(double x)
    {
    	int i;
    	for(i=1;i<=n<<1;i++)	v[i]=A[i]-x,s[i]=s[i-1]+v[i];
    	h1=h2=t1=1,t2=0,q1[1]=0;
    	for(i=L;i<=n<<1;i++)
    	{
    		while(h1<=t1&&q1[h1]<i-R)	h1++;
    		while(h2<=t2&&q2[h2]<i-R)	h2++;
    		if(!(i&1)&&h1<=t1&&s[q1[h1]]<=s[i])
    		{
    			ans1=S[i]-S[q1[h1]],ans2=i-q1[h1],g=gcd(ans1,ans2),ans1/=g,ans2/=g;
    			return 1;
    		}
    		if((i&1)&&h2<=t2&&s[q2[h2]]<=s[i])
    		{
    			ans1=S[i]-S[q2[h2]],ans2=i-q2[h2],g=gcd(ans1,ans2),ans1/=g,ans2/=g;
    			return 1;
    		}
    		if(!((i-L+1)&1))
    		{
    			while(h1<=t1&&s[q1[t1]]>=s[i-L+1])	t1--;
    			q1[++t1]=i-L+1;
    		}
    		else
    		{
    			while(h2<=t2&&s[q2[t2]]>=s[i-L+1])	t2--;
    			q2[++t2]=i-L+1;
    		}
    	}
    	return 0;
    }
    inline int rd()
    {
    	int ret=0,f=1;	char gc=getchar();
    	while(gc<'0'||gc>'9')	{if(gc=='-')f=-f;	gc=getchar();}
    	while(gc>='0'&&gc<='9')	ret=ret*10+gc-'0',gc=getchar();
    	return ret*f;
    }
    int main()
    {
    	n=rd(),L=rd(),R=rd();
    	int i;
    	double l=1<<30,r=0,mid;
    	for(i=1;i<=n;i++)	A[i]=A[i+n]=rd(),l=min(l,(double)A[i]),r=max(r,(double)A[i]);
    	for(i=1;i<=n<<1;i++)	S[i]=S[i-1]+A[i];
    	for(i=1;i<=50;i++)
    	{
    		mid=(l+r)/2;
    		if(check(mid))	l=mid;
    		else	r=mid;
    	}
    	printf("%lld/%lld",ans1,ans2);
    	return 0;
    }

     

  • 相关阅读:
    flutter 右滑返回上一页
    flutter 的Animation简单了解
    Flutter Offstage、Visibility隐藏/可见
    flutter手势
    Flutter生命周期
    flutter 路由动画
    flutter 保持页面状态
    flutter 不规则底部工具栏实现
    flutter 主页面底部导航栏实现以及主题风格设置
    flutter DropdownButton使用
  • 原文地址:https://www.cnblogs.com/CQzhangyu/p/7536452.html
Copyright © 2011-2022 走看看