zoukankan      html  css  js  c++  java
  • LG P1912 [NOI2009]诗人小G

    Description

    小 G 是一个出色的诗人,经常作诗自娱自乐。但是,他一直被一件事情所困扰,那就是诗的排版问题。

    一首诗包含了若干个句子,对于一些连续的短句,可以将它们用空格隔开并放在一行中,注意一行中可以放的句子数目是没有限制的。小 G 给每首诗定义了一个行标准长度(行的长度为一行中符号的总个数),他希望排版后每行的长度都和行标准长度相差不远。显然排版时,不应改变原有的句子顺序,并且小 G 不允许把一个句子分在两行或者更多的行内。在满足上面两个条件的情况下,小 G 对于排版中的每行定义了一个不协调度, 为这行的实际长度与行标准长度差值绝对值的 $P$ 次方,而一个排版的不协调度为所有行不协调度的总和。

    小 G 最近又作了几首诗,现在请你对这首诗进行排版,使得排版后的诗尽量协调(即不协调度尽量小),并把排版的结果告诉他。

    Solution

    对于前$i$个诗句,选择的上一个换行点具有决策单调性($s$为诗句长度前缀和)

    $$dp_i=min_{j < i}(dp_j+|s_i-s_j-l+1|^p)$$

    单调队列维护递增的最优转移点

    #pragma GCC optimize(2)
    #include<iostream>
    #include<cstring>
    #include<cstdio>
    #include<deque>
    #include<cmath>
    using namespace std;
    int T,n,l,p,s[100005],g[100005],r[100005],tot;
    char c[100005][35];
    long double dp[100005];
    deque<int>qpos,q;
    inline int read()
    {
        int w=0,f=1;
        char ch=0;
        while(ch<'0'||ch>'9')
        {
            if(ch=='-')
                f=-1;
            ch=getchar();
        }
        while(ch>='0'&&ch<='9')
        {
            w=(w<<1)+(w<<3)+ch-'0';
            ch=getchar();
        }
        return w*f;
    }
    long double ksm(long double a,int P)
    {
        long double ret=1;
        while(P)
        {
            if(P&1)
            {
                ret*=a;
            }
            a*=a;
            P>>=1;
        }
        return ret;
    }
    long double cal(int i,int j)
    {
        return dp[j]+ksm((long double)abs(s[i]-s[j]-l-1),p);
    }
    int find(int x,int y)
    {
        int l=x,r=n+1,ans=r;
        while(l<=r)
        {
            int mid=l+r>>1;
            if(cal(mid,x)<cal(mid,y))
            {
                l=mid+1;
            }
            else
            {
                ans=mid;
                r=mid-1;
            }
        }
        return ans;
    }
    int main()
    {
        T=read();
        for(;T;T--)
        {
            n=read();
            l=read();
            p=read();
            for(int i=1;i<=n;i++)
            {
                scanf("%s",c[i]);
                s[i]=s[i-1]+strlen(c[i])+1;
            }
            q.clear();
            qpos.clear();
            q.push_back(0);
            qpos.push_back(0);
            for(int i=1;i<=n;i++)
            {
                while(q.size()>1&&qpos[1]<=i)
                {
                    q.pop_front();
                    qpos.pop_front();
                }
                g[i]=q.front();
                dp[i]=cal(i,g[i]);
                while(q.size()>1&&qpos.back()>=find(q.back(),i))
                {
                    qpos.pop_back();
                    q.pop_back();
                }
                qpos.push_back(find(q.back(),i));
                q.push_back(i);
            }
            if(dp[n]>1e18)
            {
                puts("Too hard to arrange");
            }
            else
            {
                printf("%lld
    ",(long long)dp[n]);
                /*for(int i=n;i;i=g[i])
                {
                    r[++tot]=g[i];
                }
                r[0]=n;
                while(tot)
                {
                    for(int i=r[tot]+1;i<r[tot-1];i++)
                    {
                        printf("%s ",c[i]);
                    }
                    puts(c[r[tot-1]]);
                    --tot;
                }*/
            }
            puts("--------------------");
        }
        return 0;
    }
    诗人小G
  • 相关阅读:
    Python基础-面向对象1
    Centos升级安装.Net core 1.1
    员工大规模离职事件的处理方法和启示
    React Redux学习笔记
    Tfs 2015 代理池配置笔记
    自动化测试UI Test, Performance Test, Load Test 总结整理
    [转]【长文干货】浅析分布式系统
    .Net身份验证概述
    Owin中间件搭建OAuth2.0认证授权服务体会
    使用Owin中间件搭建OAuth2.0认证授权服务器
  • 原文地址:https://www.cnblogs.com/JDFZ-ZZ/p/13493747.html
Copyright © 2011-2022 走看看