https://www.luogu.com.cn/problem/CF1702E

转化题意

把所有数连边,判断是否为二分图。

染色法

void solve() {
    #define tests
    int n;
    std::cin >> n;
    std::map<int, std::vector<int>> edge;
    std::vector<bool> used(n + 1);
    bool ok(true);
    for (int i = 0; i < n; i++) {
        int a, b;
        std::cin >> a >> b;
        edge[a].push_back(b); 
        edge[b].push_back(a);
        if (a == b or sz(edge[a]) > 2 or sz(edge[b]) > 2) {ok = false;}//出现次数大于二
    }

    if (not ok) {NO; return ;}

    auto dfs = [&](auto self, int now) -> int {
        used[now] = true;
        for (auto to : edge[now]) if (not used[to]) {
            return self(self, to) + 1;
        }
        return 1;
    };//其实也是判断有没有奇环,显然总数加起来如果是奇数就是奇环

    for (int i = 0; i < n; i++) {
        if (not used[i + 1] and (dfs(dfs, i + 1) & 1)) {
            ok = false;
        }
    }
    ok ? YES : NO;
}

扩展域并查集

//扩展域并查集维护是否为二分图
//构造出x的敌人x+n和y的敌人y+n
//因为是分成两个集合,所以只要构造一个敌人(即另一个与自己相等的数,但不能放到同一个集合内所以称为敌人)
//把x和y的敌人合并,y和x的敌人合并
//如果x和自己的敌人在一个并查集,就不合法

void solve() {
    #define tests
    int n;
    std::cin >> n;
    std::vector<int> cnt(n);
    DSU dsu(n * 2);
    for (int i = 0; i < n; i++) {
        int x, y;
        std::cin >> x >> y;
        x--, y--;
        cnt[x]++; cnt[y]++;
        dsu.merge(x, y + n); dsu.merge(x + n, y);
    }
    bool ok(true);
    for (int i = 0; i < n; i++) {
        if (cnt[i] > 2) ok = false;
        if (dsu.find(i) == dsu.find(i + n)) ok = false;
    }
    ok ? YES : NO;
}
声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。