• bzoj 4709: [Jsoi2011]柠檬


    Description

    Flute 很喜欢柠檬。它准备了一串用树枝串起来的贝壳,打算用一种魔法把贝壳变成柠檬。贝壳一共有 N (1 ≤ N
    ≤ 100,000) 只,按顺序串在树枝上。为了方便,我们从左到右给贝壳编号 1..N。每只贝壳的大小不一定相同,
    贝壳 i 的大小为 si(1 ≤ si ≤10,000)。变柠檬的魔法要求,Flute 每次从树枝一端取下一小段连续的贝壳,并
    选择一种贝壳的大小 s0。如果 这一小段贝壳中 大小为 s0 的贝壳有 t 只,那么魔法可以把这一小段贝壳变成 s
    0t^2 只柠檬。Flute 可以取任意多次贝壳,直到树枝上的贝壳被全部取完。各个小段中,Flute 选择的贝壳大小 s
    0 可以不同。而最终 Flute 得到的柠檬数,就是所有小段柠檬数的总和。Flute 想知道,它最多能用这一串贝壳
    变出多少柠檬。请你帮忙解决这个问题。

    Solution

    首先猜一个结论:每一段的开头和结尾的颜色是一样的,且这一段选择的颜色一定就是开头的颜色
    这样就可以得到DP式 (f[i]=min(f[j-1]+a[i]*(s[i]-s[j]+1)^2),a[i]==a[j])
    这个东西是有决策单调性的,因为平方函数增长快,所以前面位置的一定到后面会越来越大
    下面的每一个数字是下标的话,大致就是这样分布的:
    333222111111

    我们每一次判断一个决策能否被另一个决策覆盖,如果能我们就弹掉这个元素,用一个单调栈维护就行了
    找分界点可以用二分求出,也可以直接压在栈里,减少常数

    #include<bits/stdc++.h>
    #define p (S[o].size()-1)
    #define q (S[o].size()-2)
    using namespace std;
    typedef long long ll;
    const int N=1e5+10;
    int n,a[N],id[N],c[N];ll f[N];
    vector<int>S[N/10];
    inline ll g(int x,int y){return f[x-1]+1ll*a[x]*y*y;}
    inline int k(int x,int y){
        int l=1,r=n,mid,ret=n+1;
        while(l<=r){
            mid=(l+r)>>1;
            if(g(x,mid-id[x]+1)>=g(y,mid-id[y]+1))ret=mid,l=mid+1;
            else r=mid-1;
        }
        return ret;
    }
    int main(){
      freopen("pp.in","r",stdin);
      freopen("pp.out","w",stdout);
      scanf("%d",&n);
      for(int i=1,o;i<=n;i++){
          scanf("%d",&a[i]);o=a[i];id[i]=++c[o];
          while(S[o].size()>=2 && k(S[o][p],S[o][q])<=k(i,S[o][p]))S[o].pop_back();
          S[o].push_back(i);
          while(S[o].size()>=2 && k(S[o][p],S[o][q])<id[i])S[o].pop_back();
          f[i]=g(S[o][p],c[o]-id[S[o][p]]+1);
      }
      cout<<f[n]<<endl;
      return 0;
    }
    
    
  • 相关阅读:
    C++ 引用的作用和用法
    const和指针
    sizeof的用法
    C++数组初始化的问题
    C++变量的存储类别(动态存储、静态存储、自动变量、寄存器变量、
    c++变量在内存中的存储区域(转)
    C语言的 &数组名 和 数组名的区别
    C语言运算中的数据类型自动转换原则
    链表反转
    常见的排序方法
  • 原文地址:https://www.cnblogs.com/Yuzao/p/8661253.html
Copyright © 2020-2023  润新知