背包+bitset

题目描述

在这里插入图片描述

题解:(参考于此

这个题是个典型的分组背包。
f[i][j] 为前i个数能否构成 j ,考虑第i个数选还是不选。
显然 f [ i-1 ][ j-x[i]2]==1的话f[i][j]就是可以的,x[i]的取值范围是 l[i] 到 r[i]。

这样的话其实有点浪费——f数组是一个只有01两种值的数组,哪怕表示成bool类型都有点多余,于是我们可以考虑用bitset来优化它。

bitset你可以理解为一个长度很长的01数字串(实际上它是用int拼接而成),也可以理解为可以用位运算的bool数组

这样的话一行就可以一起求——如果我们用 f[i] 来表示第i行的01串:那么 f[i] = f [i] ( f[ i-1 ] << ( x[j]2 ) )

然后bitset自带一个求1的个数的函数count,这就非常美滋滋了。

特别说明:bitset内部的实现其实是用int拼接形成所以时间复杂度是O(长度/32)的,那么本题的时间复杂度是O(n2/32)。

AC Code

#include<bits/stdc++.h>
using namespace std;
const int N=1e6+10;
bitset<N>a,b;
int main()
{
    int n;
    scanf("%d",&n);
    b[0]=1;
    for(int i=1;i<=n;i++)
    {
        int l,r;
        scanf("%d%d",&l,&r);
        for(int j=l;j<=r;j++)   //能取的数是l到r
            a=a|b<<(j*j);
        b=a;
        a.reset();
    }
    printf("%d\n",b.count());
    system("pause");
    return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值