zoukankan      html  css  js  c++  java
  • 【Codeforces】Gym 101173B Bipartite Blanket 霍尔定理+状压DP

    题意

    给一张$n imes m$二分图,带点权,问有多少完美匹配子集满足权值和大于等于$t$


    这里有一个结论:对于二分图$mathbb{A}$和$mathbb{B}$集合,如果子集$A in mathbb{A},B in mathbb{B}$,且$A,B$分别是完美匹配的子集,那么$A cup B$属于一个完美匹配

    有了这个结论之后,考虑单侧,枚举子集$S$,利用霍尔定理判定$S$是否是完美匹配,并通过dp转移状态,记录下单侧所有满足条件的权值和,然后两侧一起考虑累加得到答案

    时间复杂度$O((n+m)2^{max(n,m)})$

    代码

    #include <bits/stdc++.h>
    using namespace std;
    typedef long long LL;
    const int N = 1 << 20;
    int n, m, a[N + 5], b[N + 5], cnt[N + 5], L[N + 5], R[N + 5], fl[N + 5], fr[N + 5], t;
    char str[100][100];
    vector<int> g1, g2;
    int main() {
        scanf("%d%d", &n, &m);
        for(int i = 0; i < n; ++i) scanf("%s", str[i]);
        for(int i = 0; i < n; ++i) scanf("%d", &a[i]);
        for(int i = 0; i < m; ++i) scanf("%d", &b[i]);
        for(int i = 0; i < n; ++i) {
            for(int j = 0; j < m; ++j) {
                if(str[i][j] == '1') {
                    L[i] |= (1 << j); R[j] |= (1 << i);
                }
            }
        }
        scanf("%d", &t);
        for(int i = 0; i <= max((1 << n), (1 << m)); ++i) cnt[i] = cnt[i>>1] + (i & 1);
        for(int s = 0; s < (1 << n); ++s) {
            int now = 0, v = 0;
            fl[s] = 1;
            for(int i = 0; i < n; ++i) {
                if((s >> i) & 1) {
                    v += a[i]; now |= L[i];
                    fl[s] &= fl[s ^ (1 << i)];
                }
            }
            if(fl[s] && cnt[s] <= cnt[now]) g1.push_back(v);
            else fl[s] = 0;
        }
        for(int s = 0; s < (1 << m); ++s) {
            int now = 0, v = 0;
            fr[s] = 1;
            for(int i = 0; i < m; ++i) {
                if((s >> i) & 1) {
                    v += b[i]; now |= R[i];
                    fr[s] &= fr[s ^ (1 << i)];
                }
            }
            if(fr[s] && cnt[s] <= cnt[now]) g2.push_back(v);
            else fr[s] = 0;
        }
        sort(g1.begin(), g1.end());
        LL ans = 0;
        for(int i = 0; i < g2.size(); ++i) {
            ans += g1.size() - (lower_bound(g1.begin(), g1.end(), t - g2[i]) - g1.begin());
        }
        cout << ans << endl;
        return 0;
    }
    /* 
    3 3
    010
    111
    010
    1 2 3
    8 5 13
    21
     */
    /* 
    3 2
    01
    11
    10
    1 2 3
    4 5
    8
     */
    
  • 相关阅读:
    OSPF 开放最短路径优先协议
    RIP 路由算法
    原创 记一个上门洗车服务范围的需求实现
    转载 一位资深程序员大牛给予Java学习者的学习路线建议
    原创 解决异步调用实时跳转
    FIFO队列 ADT接口 数组实现
    FIFO队列 ADT接口 链表实现
    约瑟夫问题 链表实现
    合并-查找接口实现
    快速查找 快速合并 加权快速合并 路径等分加权快速合并 算法
  • 原文地址:https://www.cnblogs.com/ogiso-setsuna/p/8455391.html
Copyright © 2011-2022 走看看