Luogu 2766 - 最长不下降子序列问题 - [LIS问题][DP+网络流]

题目链接:https://www.luogu.org/problemnew/show/P2766

 

题解(大量参考https://blog.csdn.net/ZscDst/article/details/82423342):

第一问,可以用DP求解,用 $f[i]$ 表示以 $a[i]$ 为结尾的最长不减子序列的长度,DP时间复杂度 $O(n^2)$,假设求得长度为 $len$。

第二问我们可以用网络流来求解:

1、由于每个点只能被选一次,所以拆点,控制每个数只能选一次。

2、源点往所有 $f[i]=1$ 的点连一条边权为 $1$ 的边,所有 $f[i]=len$ 的点往汇点连一条边权为 $1$ 的边。

3、如果 j 点是由 i 点转移得到的(即f[i]+1==f[j] && a[i] <= a[j]),那么 i+n 往 j 连一条边权为1的边。

第三问:

$x[1]$ 和 $x[n]$ 可以使用多次,首先 $x[1],x[n]$ 拆出来的两个点之间的容量需要修改。

同时,此题中必然有一条边从源点连向 $x[1]$,所以这条边的容量也要修改。$x[n]$ 不一定是汇点,若 $x[n]$ 是汇点,那么 $x[n]$ 到汇点的边的容量也要修改。

如果重新构图去跑可能会超时,我们知道网络流是可以继续在残量网络中跑的,所以我们直接再添加新边,再继续跑网络流,累加到第二问的答案上,即为第三问的答案。

 

AC代码:

#include
#define I(x) x
#define O(x) (n+x)
using namespace std;
const int INF=0x3f3f3f3f;
const int maxn=505;

int n,x[maxn];
int f[maxn];

struct Edge{
    int u,v,c,f;
};
struct Dinic
{
    static const int SIZE=2*maxn;
    int s,t; //源点汇点
    vector E;
    vector<int> G[SIZE];
    void init(int l,int r)
    {
        E.clear();
        for(int i=l;i<=r;i++) G[i].clear();
    }
    void addedge(int from,int to,int cap)
    {
        E.push_back((Edge){from,to,cap,0});
        E.push_back((Edge){to,from,0,0});
        G[from].push_back(E.size()-2);
        G[to].push_back(E.size()-1);
    }
    int dist[SIZE],vis[SIZE];
    queue<int> q;
    bool bfs() //在残量网络上构造分层图
    {
        memset(vis,0,sizeof(vis));
        while(!q.empty()) q.pop();
        q.push(s);
        dist[s]=0;
        vis[s]=1;
        while(!q.empty())
        {
            int now=q.front(); q.pop();
            for(int i=0;i)
            {
                Edge& e=E[G[now][i]]; int nxt=e.v;
                if(!vis[nxt] && e.c>e.f)
                {
                    dist[nxt]=dist[now]+1;
                    q.push(nxt);
                    vis[nxt]=1;
                }
            }
        }
        return vis[t];
    }
    int dfs(int now,int flow)
    {
        if(now==t || flow==0) return flow;
        int rest=flow,k;
        for(int i=0;rest>0 && i)
        {
            Edge &e=E[G[now][i]]; int nxt=e.v;
            if(e.c>e.f && dist[nxt]==dist[now]+1)
            {
                k=dfs(nxt,min(rest,e.c-e.f));
                if(!k) dist[nxt]=0; //剪枝,去掉增广完毕的点
                e.f+=k; E[G[now][i]^1].f-=k;
                rest-=k;
            }
        }
        return flow-rest;
    }
    int mf; //存储最大流
    int maxflow()
    {
        mf=0;
        int flow=0;
        while(bfs()) while(flow=dfs(s,INF)) mf+=flow;
        return mf;
    }
}dinic;

int main()
{
    cin>>n;
    for(int i=1;i<=n;i++) scanf("%d",&x[i]);

    int len=1;
    dinic.init(dinic.s=0,dinic.t=2*n+1);
    for(int i=1;i<=n;i++)
    {
        dinic.addedge(I(i),O(i),1);

        f[i]=1;
        for(int j=1;j)
            if(x[j]<=x[i]) f[i]=max(f[i],f[j]+1);
        len=max(len,f[i]);

        if(f[i]==1) dinic.addedge(dinic.s,I(i),1);
        else
        {
            for(int j=1;j)
                if(x[j]<=x[i] && f[j]+1==f[i]) dinic.addedge(O(j),I(i),1);
        }
    }
    cout<endl;

    for(int i=1;i<=n;i++)
        if(f[i]==len) dinic.addedge(O(i),dinic.t,1);

    int ans=dinic.maxflow();
    cout<endl;


    dinic.addedge(dinic.s,I(1),INF), dinic.addedge(I(1),O(1),INF);
    dinic.addedge(I(n),O(n),INF);
    if(f[n]==len) dinic.addedge(O(n),dinic.t,INF);

    cout<endl;
}

 

你可能感兴趣的:(Luogu 2766 - 最长不下降子序列问题 - [LIS问题][DP+网络流])