APIO2015巴厘岛的雕塑——数位DP

题目:https://www.luogu.org/problemnew/show/P3646

对于A>1,将答案各位全置1,然后从高位到低位改成0判断是否可行;

用f[i][j]数组代表前i个数分成j组是否可行,转移是枚举最后一段的左端点k,然后看看后面整个一段的和能否满足要求,如果前后都满足就表示i,j状态也可行;

对于A=1,可以贪心地认为分组数量越少越好,所以可行性转化为最优性,省去一维,转移条件同上,取min即可;

先写了个WA一半的版本:

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
typedef long long ll;
int n,A,B,len;
ll f2[2005],ans,s[2005];
bool f[105][105];//可行性
bool dp1(ll x)
{
    memset(f,0,sizeof f);
    f[0][0]=1;
    for(int i=1;i<=n;i++)//前i个数分成j段 <- 前k个数分成j-1段
        for(int j=1;j<=i;j++)
            for(int k=0;k<i;k++)//0
                if(((s[i]-s[k])|x)==x)f[i][j]|=f[k][j-1];
    for(int i=A;i<=B;i++)
        if(f[n][i])return 1;
    return 0;
}
bool dp2(ll x)
{
//    memset(f2,0x3f,sizeof f2);
    f2[0]=0;
    for(int i=1;i<=n;i++)
    {
        ll ad=n+1;
        for(int j=0;j<i;j++)//0
            if(((s[i]-s[j])|x)==x)ad=min(ad,f2[j]);
        f2[i]=ad+1;
    }
    return f2[n]<=B;
}
int main()
{
    scanf("%d%d%d",&n,&A,&B);
    for(int i=1;i<=n;i++)
        scanf("%lld",&s[i]);
    for(int i=1;i<=n;i++)
        s[i]+=s[i-1];
    for(len = 0;(1LL << len) <= s[n];len++);len--;//位数
    if(A!=1)
    {
        ans=(ll)(1<<(len+1));ans--;
        for(int k=len;k>=0;k--)//0!
        {
            ll tmp=ans-(ll)(1<<k);
            if(dp1(tmp))ans=tmp;
        }
    }
    else
    {
        ans=(ll)(1<<(len+1));ans--;
        for(int k=len;k>=0;k--)
        {
            ll tmp=(ll)ans-(1<<k);
            if(dp2(tmp))ans=tmp;
        }
    }
    printf("%lld",ans);
    return 0;
}

后来又直接改成别的写法A的,但还是不太明白原来的写法为什么不行,有什么不同。

代码如下:

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
typedef long long ll;
int n,A,B,len;
ll f2[2005],ans,s[2005];
bool f[105][105];//可行性
ll dp1()
{
    ans=0;
    for(int t=len;t>=0;t--)
    {
        ans+=(1LL<<t)-1;
        memset(f,0,sizeof f);
        f[0][0]=1;
        for(int i=1;i<=n;i++)//前i个数分成j段 <- 前k个数分成j-1段
            for(int j=1;j<=i;j++)
                for(int k=0;k<i;k++)//0
                    if(((s[i]-s[k])|ans)==ans)f[i][j]|=f[k][j-1];
        bool fl=0;
        for(int i=A;i<=B;i++)fl|=f[n][i];
        if(fl)ans-=(1LL<<t)-1;
        else ans++;
    }
    return ans;
}
ll dp2()
{
    ans=0;
    for(int t=len;t>=0;t--)
    {
        ans+=(1LL<<t)-1;
        f2[0]=0;
        for(int i=1;i<=n;i++)
        {
            ll ad=n+1;
            for(int j=0;j<i;j++)//0
                if(((s[i]-s[j])|ans)==ans)ad=min(ad,f2[j]);
            f2[i]=ad+1;
        }
        if(f2[n]<=B)ans-=(1LL<<t)-1;
        else ans++;
    }
    return ans;
}
int main()
{
    scanf("%d%d%d",&n,&A,&B);
    for(int i=1;i<=n;i++)
        scanf("%lld",&s[i]);
    for(int i=1;i<=n;i++)
        s[i]+=s[i-1];
    for(len = 0;(1LL << len) <= s[n];len++);len--;//位数
    if(A==1)printf("%lld",dp2());
    else printf("%lld",dp1());
    return 0;
}

原文地址:https://www.cnblogs.com/Zinn/p/8971746.html

时间: 2024-10-01 05:01:29

APIO2015巴厘岛的雕塑——数位DP的相关文章

【BZOJ4069】[Apio2015]巴厘岛的雕塑 按位贪心+DP

[BZOJ4069][Apio2015]巴厘岛的雕塑 Description 印尼巴厘岛的公路上有许多的雕塑,我们来关注它的一条主干道. 在这条主干道上一共有 N 座雕塑,为方便起见,我们把这些雕塑从 1 到 N 连续地进行标号,其中第 i 座雕塑的年龄是 Yi 年.为了使这条路的环境更加优美,政府想把这些雕塑分成若干组,并通过在组与组之间种上一些树,来吸引更多的游客来巴厘岛. 下面是将雕塑分组的规则: 这些雕塑必须被分为恰好 X 组,其中 A< = X< = B,每组必须含有至少一个雕塑,每

bzoj 4069 [Apio2015]巴厘岛的雕塑 dp

[Apio2015]巴厘岛的雕塑 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 494  Solved: 238[Submit][Status][Discuss] Description 印尼巴厘岛的公路上有许多的雕塑,我们来关注它的一条主干道. 在这条主干道上一共有 N 座雕塑,为方便起见,我们把这些雕塑从 1 到 N 连续地进行标号,其中第 i 座雕塑的年龄是 Yi 年.为了使这条路的环境更加优美,政府想把这些雕塑分成若干组,并通过在组与组之间

【BZOJ 4069】[apio2015]巴厘岛的雕塑

4069:[apio2015]巴厘岛的雕塑 Time limit: 1000 ms Memory limit: 65536 KB Description The province of Bali has many sculptures located on its roads. Let's focus on one of its main roads. There are N sculptures on that main road, conveniently numbered 1 throug

bzoj千题计划239:bzoj4069: [Apio2015]巴厘岛的雕塑

http://www.lydsy.com/JudgeOnline/problem.php?id=4069 a!=1: 从高位到低位一位一位的算 记录下哪些位必须为0 dp[i][j] 表示前i个数分为j组,这一位为0,且满足之前必须为0的位也是0 是否可行 枚举k,表示k+1~i分为一组 若k+1~i的和满足 必须为0的位是0,且dp[k][j-1] 为true 则dp[i][j]为true a=1: 从高位到低位一位一位的算 记录下哪些位必须为0 dp[i] 表示前i个数,这一位为0,且满足之

[APIO2015]巴厘岛的雕塑[按位贪心+dp]

题意 给你长度为 \(n\) 的序列,要求分成 \(k\) 段连续非空的区间,求所有区间和的 \(or\) 最小值. 分析 定义 \(f_{i,j}\) 表示前 \(i\) 个点分成 \(j\) 段的最小 \(or\) 是有问题的,因为可能有一位一定在后面出现而某一位没有必要在后面出现,这时贪心就出现了问题. 考虑按位贪心,假设考虑到了第 \(b\) 位,定义 \(f_{i,j}\) 表示前 \(i\) 个点分成 \(j\) 段且满足 \(b\) 以上位的限制,第 \(b\) 位为 \(0\)

4069: [Apio2015]巴厘岛的雕塑

Description 印尼巴厘岛的公路上有许多的雕塑,我们来关注它的一条主干道. 在这条主干道上一共有 N 座雕塑,为方便起见,我们把这些雕塑从 1 到 N 连续地进行标号,其中第 i 座雕塑的年龄是 Yi 年.为了使这条路的环境更加优美,政府想把这些雕塑分成若干组,并通过在组与组之间种上一些树,来吸引更多的游客来巴厘岛. 下面是将雕塑分组的规则: 这些雕塑必须被分为恰好 X 组,其中 A< = X< = B,每组必须含有至少一个雕塑,每个雕塑也必须属于且只属于一个组.同一组中的所有雕塑必须

[APIO2015]巴厘岛的雕塑

刚开始看错题,以为是每一组或起来,求和的最小值,然后就瞎写了个dp,状态根本无法转移竟然过了六个点也是醉了 这个题根据数据可以分成两部分,可以按位贪心检验 对于n<=100的数据每一位直接n^3的dp就可以了 然后对于n>100的数据,A相当于没有限制,对每一位可以直接搞, g[i]代表到第i个雕塑至少分多少组才能符合要求 1 #define MAXN 2010UL 2 #include <cstdio> 3 #include <cstring> 4 5 using n

bzoj4069: [Apio2015]巴厘岛的雕塑

暴力三维DP容易想到 也容易想到贪心每一位,尽量不选 但是怎么check呢?又是拼起来 f[i][j]表示枚举到第i位,分成j组,且可以满足当前假设是否有可行解 转移不难自己看代码吧 但是这样是O(logm*n^3)的,subtest5过不去 但是它又有一个A==1,就可以维护枚举到第i位分成组数最小的方案数 好题啊! #include<cstdio> #include<iostream> #include<cstring> #include<cstdlib>

bzoj4069【APIO2015】巴厘岛的雕塑

4069: [Apio2015]巴厘岛的雕塑 Time Limit: 10 Sec  Memory Limit: 64 MB Submit: 192  Solved: 89 [Submit][Status][Discuss] Description 印尼巴厘岛的公路上有许多的雕塑,我们来关注它的一条主干道. 在这条主干道上一共有 N 座雕塑,为方便起见,我们把这些雕塑从 1 到 N 连续地进行标号,其中第 i 座雕塑的年龄是 Yi 年.为了使这条路的环境更加优美,政府想把这些雕塑分成若干组,并通