题意:
解法:
令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;
}