CF1399F Yet Another Segments Subset

首先注意一下题面要求,使得选出的线段两两要么包含要么不相交,也就是说一条线段可能会出现不相交的几条线段,而这些线段上面也可能继续这样包含线段。然后我们可以发现我们要做的实际上是在这条线段上选取几条线段然后递归求出子问题,这是一个 (dp) 的形式,令 (f_i) 表示在线段 (i) 上最多能选出多少个满足条件的区间。但是在这里拓扑序我们还未知,我们看看能否求出这个 (dp) 的拓扑序。我们发现对于任意一条线段我们能选择被它包含的线段必然左端点大于该线段左端点,右端点大于该线段左端点,那么我们只需要按照右端点排序,那么每次能转移到的区间就都会已经计算完毕。但需要注意的是如果右端点相同,我们还要按照左端点从大到小排序,这样才能保证小的线段先算。

接下来考虑如何转移,实际上这里需要求解的一个问题就是每个线段有一个权值,选出一些不相交的线段使得它们的权值最大。接下来可以发现如果当前线段能选,那么上次选的最右边线段的右端点一定要比当前线段的左端点要靠前,于是可以令 (dp_i) 表示当前选择的最右边的右端点在 (i) 的最大权值,有转移:

[dp_{a_i.r} = max{dp_{a_j.r} + f_i}(a_j.r < a_i.l) ]

(f_i) 拉出来这就是一个前缀 (max) 的形式,因为右端点是单调递增的,因此只需要沿途维护一个指针扫过来并统计即可。本题当中值域很大?可以发现我们只关注大小关系,离散化即可。

#include<bits/stdc++.h>
using namespace std;
#define rep(i, l, r) for(int i = l; i <= r; ++i)
const int N = 6000 + 5;
struct node{
    int l, r;
}a[N];
int T, n, tot, cnt, f[N], g[N], d[N], dp[N];
int read(){
    char c; int x = 0, f = 1;
    c = getchar();
    while(c > '9' || c < '0'){ if(c == '-') f = -1; c = getchar();}
    while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
    return x * f;
}
bool cmp(node a, node b){
    return a.r == b.r ? a.l > b.l : a.r < b.r;
}
int main(){
    T = read();
    while(T--){
        n = read(), tot = 0;
        rep(i, 1, n) a[i].l = read(), a[i].r = read(), d[++tot] = a[i].l, d[++tot] = a[i].r;
        sort(d + 1, d + tot + 1), sort(a + 1, a + n + 1, cmp);
        cnt = unique(d + 1, d + tot + 1) - d - 1;
        rep(i, 1, n){
            a[i].l = lower_bound(d + 1, d + cnt + 1, a[i].l) - d;
            a[i].r = lower_bound(d + 1, d + cnt + 1, a[i].r) - d;
        }
        rep(i, 1, n){
            int P = 0, l = 0; 
            rep(j, 1, a[i - 1].r) dp[j] = g[j] = 0;
            rep(j, 1, i - 1) if(a[j].l >= a[i].l){
                if(a[j].r != l) ++P;
                while(P < a[j].r) g[P] = g[P - 1], ++P;
                dp[a[j].r] = max(dp[a[j].r], g[a[j].l - 1] + f[j]), g[P] = max(g[P - 1], dp[a[j].r]);
                l = a[j].r;
            }
            f[i] = g[P] + 1;
        }
        int P = 0, l = 0;
        rep(i, 1, a[n].r) dp[i] = g[i] = 0; 
        rep(i, 1, n){
            if(a[i].r != l) ++P;
            while(P < a[i].r) g[P] = g[P - 1], ++P;
            dp[a[i].r] = max(dp[a[i].r], g[a[i].l - 1] + f[i]), g[P] = max(g[P - 1], dp[a[i].r]);
            l = a[i].r;
        }
        printf("%d
", g[a[n].r]);
        rep(i, 1, a[n].r) dp[i] = f[i] = g[i] = 0;
    }
    return 0;
}
GO!
原文地址:https://www.cnblogs.com/Go7338395/p/13620810.html