zoukankan      html  css  js  c++  java
  • 「Luogu3357」 最长k可重线段集问题

    「Luogu3357」 最长k可重线段集问题

    problem

    Solution

    「Luogu3357」 最长k可重区间集问题类似,但此题需要考虑斜率不存在的线段

    我们将每个线段的两个端点中(x)坐标较小的那一个认为是线段的起点,另一个为终点

    考虑拆点,我们将坐标上的每一个点拆成两个点(2*x,2*x+1)。对于一条线段,如果(x)是它的起点,将它设为(2*x+1),否则设为(2*x),这样我们就把每个坐标拆成了两个类似“出点”和“入点”的东西

    剩下的连边方式和区间没有区别

    Code

    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    #include <iostream>
    #include <cstdlib>
    #include <cmath>
    #include <queue>
    #define maxn 25005
    using namespace std;
    typedef long long ll;
    
    template <typename T>void read(T &t)
    {
        t=0;char c=getchar();ll f=0;
        while(!isdigit(c)){f|=c=='-';c=getchar();}
        while(isdigit(c)){t=t*10+c-'0';c=getchar();}
        if(f)t=-t;
    }
    
    const ll inf=0x3f3f3f3f3f3f3f3f;
    ll n,K;
    ll xa[maxn],xb[maxn],z[maxn];
    ll s,t;
    ll ansc;
    
    struct edge
    {
        ll u,v,f,c,nxt;
    }g[maxn];
    
    ll head[maxn],ecnt=1;
    void eADD(ll u,ll v,ll f,ll c)
    {
        g[++ecnt].u=u;
        g[ecnt].v=v;
        g[ecnt].f=f;
        g[ecnt].c=c;
        g[ecnt].nxt=head[u];
        head[u]=ecnt;
    }
    
    inline ll getz(ll xa,ll xb,ll ya,ll yb)
    {
        return sqrt((xa-xb)*(xa-xb)+(ya-yb)*(ya-yb));
    }
    
    ll dist[maxn],inq[maxn],minf[maxn];
    ll pree[maxn],prev[maxn];
    bool SPFA()
    {
        memset(dist,0x3f,sizeof(dist));
        memset(minf,0x3f,sizeof(minf));
        queue<int> q;
        dist[s]=0;
        inq[s]=1;
        q.push(s);
        while(!q.empty())
        {
            ll u=q.front();
            q.pop();
            inq[u]=0;
            for(register ll i=head[u];i;i=g[i].nxt)
            {
                ll v=g[i].v;
                if(g[i].f && dist[v]>dist[u]+g[i].c)
                {
                    prev[v]=u,pree[v]=i;
                    dist[v]=dist[u]+g[i].c;
                    minf[v]=min(minf[u],g[i].f);
                    if(!inq[v])
                    {
                        inq[v]=1;
                        q.push(v);
                    }
                }
            }
        }
        return dist[t]<inf;
    }
    
    int main()
    {
        read(n),read(K);
        ll ocr[maxn<<1];
        for(register ll i=1;i<=n;++i)
        {
            ll ya,yb;
            read(xa[i]),read(ya),read(xb[i]),read(yb),z[i]=getz(xa[i],xb[i],ya,yb);
            xa[i]<<=1,xb[i]<<=1;
            if(xa[i]>xb[i])swap(xa[i],xb[i]);
            if(xa[i]==xb[i])xb[i]|=1;
            else xa[i]|=1;
            ocr[++ocr[0]]=xa[i],ocr[++ocr[0]]=xb[i];
        }
        sort(ocr+1,ocr+ocr[0]+1);
        ocr[0]=unique(ocr+1,ocr+ocr[0]+1)-ocr-1;
        for(register ll i=1;i<=n;++i)
            xa[i]=lower_bound(ocr+1,ocr+ocr[0]+1,xa[i])-ocr,xb[i]=lower_bound(ocr+1,ocr+ocr[0]+1,xb[i])-ocr;
        s=0,t=ocr[0]+1;
        eADD(s,1,K,0),eADD(1,s,0,0);
        eADD(ocr[0],t,K,0),eADD(t,ocr[0],0,0);
        for(register ll i=1;i<ocr[0];++i)
            eADD(i,i+1,inf,0),eADD(i+1,i,0,0);
        for(register ll i=1;i<=n;++i)
            eADD(xa[i],xb[i],1,-z[i]),eADD(xb[i],xa[i],0,z[i]);
        while(SPFA())
        {
            ansc+=dist[t]*minf[t];
            for(register ll i=t;i!=s;i=prev[i])
            {
                g[pree[i]].f-=minf[t];
                g[pree[i]^1].f+=minf[t];
            }
        }
        printf("%lld",-ansc);
        return 0;
    }
    
  • 相关阅读:
    lua学习笔记(一)
    使用npm安装一些包失败了的看过来(npm国内镜像介绍)
    ubuntu和win7 64双系统 安装
    virtualbox共享文件夹
    今天来谈谈三大基础排序选择排序、冒泡排序、插入排序
    vue入门七之vuex的使用
    简单介绍下js的随机数的生成
    来简单谈谈JavaScript两个数的交换问题
    VUE入门六之过滤器
    VUE入门五之路由的使用
  • 原文地址:https://www.cnblogs.com/lizbaka/p/10505670.html
Copyright © 2011-2022 走看看