zoukankan      html  css  js  c++  java
  • HDU多校G4

    补到两道十分有印象的题目。

    Contest of Rope Pulling

    这题做法很巧妙。

    题目大意:给定a,b两个数组,分别有n和m组的(wi,vi)。要求从两个数组中分别选出一个子数组,使得两个子数组的wi和相同,并使总的vi之和最大。

    首先一看是有关于价值和容量,可以想到用背包解题。但是两个背包建起来复杂度会到达O(1000*1000*(n+m)*T)会超时。于是我们可以想办法合并一个背包,把b数组的wi都变为负数,那么如果两个子数组的wi会相同,则它们的答案会存储在dp[0],然而如果按照先a数组再b数组这样子去遍历,这个复杂度也还是可能会超时。

    这时,一个十分巧妙的操作,将ab数组合起来随机化一遍!这样子操作,不会像先a数组再b数组遍历这种,造成全部 wi 相加过大,要遍历的范围过多从而超时的情况。

    然而不知什么原因,无限RE……

    Go Running

    这题做法没想到竟然用网络流做。

    把当前的位置和时刻当作坐标系上的一个点,如果往左跑或往右跑,即相当于作一条斜率为-1或1的直线,如果一条直线交于两个点,说明这两个位置可能是同一个人跑出来的。

    于是一个点要么是属于斜率为-1的直线①,要么属于斜率为1的直线②。

    建图,直线①直线②之间连一条容量为1的边,各自连接超级源点与汇点的点之间也要连一条容量为1的边,于是这题就变成了二分图匹配问题。

    #include <bits/stdc++.h>
    #define debug freopen("r.txt","r",stdin)
    #define mp make_pair
    #define ri register int
    #define pb push_back
    using namespace std;
    typedef long long ll;
    typedef pair<int, int> pii;
    const int maxn = 1e6+10;
    const int INF = 0x3f3f3f3f; 
    const int mod = 1e9+7;
    const int N=1e6+10;
    inline ll read(){ll s=0,w=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}
    while(ch>='0'&&ch<='9') s=s*10+ch-'0',ch=getchar();
    return s*w;}
    ll qpow(ll p,ll q){return (q&1?p:1)*(q?qpow(p*p%mod,q/2):1)%mod;}
    struct FLOW{
        struct edge{int to,w,nxt;};
        vector<edge> a; int head[N];
        int cur[N];
        int n,s,t;
        queue<int> q;
        int dep[N],gap[N];
        void ae(int x,int y,int w){
            a.push_back({y,w,head[x]});
            head[x]=a.size()-1;
        }
        bool bfs(){
            fill(dep,dep+n,-1); dep[t]=0;
            fill(gap,gap+n,0); gap[0]=1;
            q.push(t);
            while(!q.empty()){
                int x=q.front(); q.pop();
                for(int i=head[x];i!=-1;i=a[i].nxt){
                    int p=a[i].to;
                    if(dep[p]!=-1)continue;
                    dep[p]=dep[x]+1;
                    q.push(p);
                    gap[dep[p]]++;
                }
            }
            return dep[s]!=-1;
        }
        int dfs(int x,int fl){
            int now,ans=0;
            if(x==t)return fl;
            for(int i=cur[x];i!=-1;i=a[i].nxt){
                cur[x]=i;
                int p=a[i].to;
                if(a[i].w && dep[p]+1==dep[x])
                if((now=dfs(p,min(fl,a[i].w)))){
                    a[i].w-=now;
                    a[i^1].w+=now;
                    ans+=now,fl-=now;
                    if(fl==0)return ans;
                }
            }
            gap[dep[x]]--;
            if(gap[dep[x]]==0)dep[s]=n;
            dep[x]++;
            gap[dep[x]]++;
            return ans;
        }
        void init(int _n){
            n=_n+1;
            a.clear();
            fill(head,head+n,-1);
        }
        int solve(int _s,int _t){ //返回最大流
            s=_s,t=_t;
            int ans=0;
            if(bfs())
            while(dep[s]<n){
                copy(head,head+n,cur);
                ans+=dfs(s,INF);
            }
            return ans;
        }
    }flow;
    void add(int x,int y,int w){flow.ae(x,y,w),flow.ae(y,x,0);}
    //先flow.init(n),再add添边,最后flow.solve(s,t).
    map<int,int> mapp1,mapp2;
    int t,n,s,i,T,x,xleft[maxn],xright[maxn],cnt1,cnt2;
    int main()
    {
        T=read();
        while (T--)
        {
            n=read();
            s=cnt1=cnt2=0;
            for (i=1;i<=n;i++)
            {
                t=read(),x=read();
                xleft[i]=x-t;
                xright[i]=x+t;
                if (!mapp1[xleft[i]]) mapp1[xleft[i]]=++cnt1;
                if (!mapp2[xright[i]]) mapp2[xright[i]]=++cnt2;
            }
            t=cnt1+cnt2+1;
            flow.init(t);
            for (i=1;i<=cnt1;i++) add(s,i,1);
            for (i=1;i<=n;i++) add(mapp1[xleft[i]],mapp2[xright[i]]+cnt1,1);
            for (i=1;i<=cnt2;i++) add(i+cnt1,t,1);
            cout<<flow.solve(s,t)<<endl;
            mapp1.clear();
            mapp2.clear();
        }
        return 0;
    }
    View Code
  • 相关阅读:
    JAVA中字符串比较equals()和equalsIgnoreCase()的区别
    JAVA字母的大小写转换
    对于java线程的理解
    JAVA实现文件导出Excel
    处理数据库中的null值问题
    POJO、JAVABean、Entity的区别
    Mybatis的choose标签使用
    redis详解
    Spring框架基础解析
    利用 BackgroundService 固定时间间隔执行某动作
  • 原文地址:https://www.cnblogs.com/Y-Knightqin/p/13642589.html
Copyright © 2011-2022 走看看