gym101908 F. Music Festival(状压dp)

题意:

在这里插入图片描述

解法:
令d[i][j]表示时间到达i,当前已选择舞台集合为二进制j的最大值.
将时间离散化,那么第一维的大小只有2e3,第二维2^10=1024,
空间总大小大概2e6.

如果存在(l,r,v,id),
那么d[l][s]可以转移到d[r][s|(1<<id)],转移收益为v.

因为(l,r,v,id)只有m个,因此最多转移m*(1<<n).

总复杂度为O(time*(1<<n)+m*(1<<n)).
time是时间离散化之后的数量,time<=2e3.
code:
#include <bits/stdc++.h>
#define int long long
using namespace std;
const int maxm=2e4+5;
struct Node{
    int l,r,v;
};
struct Node2{
    int r,v,id;
};
vector<Node>g[maxm];
vector<Node2>temp[maxm];
int xx[maxm],num;
int d[2005][1033];
int n;
void solve(){
    cin>>n;
    for(int i=0;i<n;i++){
        int k;cin>>k;
        for(int j=1;j<=k;j++){
            int l,r,v;cin>>l>>r>>v;
            g[i].push_back({l,r,v});
            xx[++num]=l;
            xx[++num]=r;
        }
    }
    sort(xx+1,xx+1+num);
    num=unique(xx+1,xx+1+num)-xx-1;
    for(int i=0;i<n;i++){
        for(auto& p:g[i]){
            p.l=lower_bound(xx+1,xx+1+num,p.l)-xx;
            p.r=lower_bound(xx+1,xx+1+num,p.r)-xx;
            temp[p.l].push_back({p.r,p.v,i});
        }
    }
    for(int i=0;i<=num;i++){
        for(int j=0;j<=(1<<n);j++){
            d[i][j]=-1e18;
        }
    }
    d[0][0]=0;
    for(int i=0;i<=num;i++){
        for(int j=0;j<(1<<n);j++){
            if(d[i][j]==-1e18)continue;
            d[i+1][j]=max(d[i+1][j],d[i][j]);
            for(auto p:temp[i]){
                d[p.r][j|(1<<p.id)]=max(d[p.r][j|(1<<p.id)],d[i][j]+p.v);
            }
        }
    }
    int ans=-1e18;
    for(int i=0;i<=num;i++){
        ans=max(ans,d[i][(1<<n)-1]);
    }
    if(ans==-1e18)ans=-1;
    cout<<ans<<endl;
}
signed main(){
    ios::sync_with_stdio(0);cin.tie(0);
    solve();
    return 0;
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值