NKOJ 3500 独立集(dp)

该博客介绍了NKOJ 3500题目的独立集问题,实际上可以转化为求最长上升子序列。博主首先解析了问题描述和输入输出格式,并提供了样例输入和输出。针对第一问,博主指出可以通过寻找最长上升子序列来解决,而第二问则提出了一种暴力求解的思路,通过维护两个集合P和Q,进行倒序讨论并删除元素,复杂度为nlogn。最后,博主提到了一种更优秀的解决方案并推荐了参考代码。
摘要由CSDN通过智能技术生成

P3500【2015多校联训6】独立集

问题描述
这里写图片描述

输入格式

输入包含两行,第一行为 N,
第二行为 1 到 N 的一个全排列

输出格式

输出包含两行,第一行输出最大独立集的大小,第二行从小到大输出一定在最大独立集 的点的编号。

样例输入

3
3 1 2

样例输出

2
2 3

提示

30%的数据满足 N<=16
60%的数据满足 N<=1,000
100%的数据满足 N<=100,000


仔细观察,发现第一问就是最长上升子序列,因为逆序对才会连边,那么没有边相连的点一定是正序的,那么问题就是求最长正序,即最长上升子序列。

考虑第二问,优秀的做法做不来,搞个暴力。
假设以 i 为结尾的最长上升子序列长度为k,那么将 A[i] 加到一个 set P 中,将i加到另一个 set , Q 中,那么跑完最长上升子序列后,令第一问答案为len

len 开始倒着讨论,用 P[k],Q[k] 中最大的元素去删除 P[k1],Q[k1] 中的元素,删除条件就是如果 P[k1],Q[k1] 中的一个元素不能转移到 P[k],Q[k] 中的任一元素,那么删除。

如果讨论时发现某 set 中仅有一个元素,那么这个元素一定要选。复杂度 nlogn

优秀做法参见THH


代码:

#include<stdio.h>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<set>
#define N 100055
using namespace std;
set<int>Q[N],R[N];
int n,A[N],F[N],B[N],id[N],ans;
bool mark[N];
int main()
{
    set<int>::iterator x,y,z;
    int i,j,k;
    scanf("%d",&n);
    for(i=1;i<=n;i++)scanf("%d",&A[i]),id[A[i]]=i;
    memset(B,60,sizeof(B));B[0]=0;Q[0].insert(0);
    for(i=1;i<=n;i++)
    {
        F[i]=lower_bound(B+1,B+n+1,A[i])-B;
        B[F[i]]=A[i];
        Q[F[i]].insert(A[i]);
        R[F[i]].insert(i);
        ans=max(ans,F[i]);
    }
    for(i=ans;i>=1;i--)
    {
        if(Q[i].size()==1)
        {
            x=Q[i].begin();
            mark[*x]=1;
        }
        x=Q[i].end();x--;
        y=Q[i-1].lower_bound(*x);
        for(z=y;z!=Q[i-1].end();z++)R[i-1].erase(R[i-1].find(id[*z]));
        Q[i-1].erase(y,Q[i-1].end());
        x=R[i].end();x--;
        y=R[i-1].lower_bound(*x);
        for(z=y;z!=R[i-1].end();z++)Q[i-1].erase(Q[i-1].find(A[*z]));
        R[i-1].erase(y,R[i-1].end());
    }
    printf("%d\n",ans);
    for(i=1;i<=n;i++)if(mark[i])printf("%d ",id[i]);
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值