Live2D

Solution -「COCI 2009-2010」「洛谷 P8076」RESTORAN

\(\mathscr{Description}\)

  Link.

  给定一个含 \(n\) 个点 \(m\) 条边的简单图, 求一种边二染色方案, 使得所有 \(\deg\ge2\) 的结点都邻接于两种颜色的边.

  \(n,m\le10^5\).

\(\mathscr{Solution}\)

  清甜味儿的题目比较调剂愚钝的大脑. (

  染色, 一开始的思路是匹配或网络流相关, 不过一看数据范围就大概弃了.

  第一个 motivation 大概也来自数据范围: 近线性做法 \(\to\) 生成树 (DFS 树) 上构造?

  很快, 我们发现树上的 "奇偶染色" (细节见最后一段描述) 是很强大的, 有且仅有树根有可能不合法. 树根在什么情况下合法呢? 简单讨论发现, 我们要求树根在一个偶环里. 可见, 无脑奇偶染色导致了一个挺严的限制因素, 这样的条件显得太奢侈啦!

  怎么办呢? 第二个 motivation 是: 保留一条特定的边, 它不在奇偶染色的考虑范围, 仅用于保证树根合法.

  保留了一条边, 树根倒是一定合法; 这条边的另一端又称了问题. 什么样的点忽略一条邻接边仍然能在奇偶染色下合法?

  其实所有 \(\deg\ge3\) 点都可以.

  因此, 我们的算法为: 找到一个 \(\deg\ge3\) 的点 \(u\), 保留其一条邻接边 \((u,v)\), 在连通块剩余边上从 \(v\) 出发 DFS, 树上奇偶染色, 返祖边和对应点子边同色. 染色完成后, 通过 \((u,v)\) 保证 \(v\) 的合法性. 此外需要特殊处理偶环. 当且仅当存在一个连通块为奇环时无解. 复杂度 \(\mathcal O(n+m)\).

\(\mathscr{Code}\)

/*+Rainybunny+*/

#include <bits/stdc++.h>

#define rep(i, l, r) for (int i = l, rep##i = r; i <= rep##i; ++i)
#define per(i, r, l) for (int i = r, per##i = l; i >= per##i; --i)

const int MAXN = 1e5;
int n, m, ecnt = 1, head[MAXN + 5], deg[MAXN + 5], clr[MAXN + 5];
struct Edge { int to, nxt; } graph[MAXN * 2 + 5];
bool vis[MAXN + 5];

inline void link(const int u, const int v) {
    graph[++ecnt] = { v, head[u] }, head[u] = ecnt;
    graph[++ecnt] = { u, head[v] }, head[v] = ecnt;
}

inline void paint(const int u, const int cur) {
    vis[u] = true;
    for (int i = head[u], v; i; i = graph[i].nxt) {
        if (!clr[i >> 1] && !vis[v = graph[i].to]) {
            // printf("%d->%d\n", u, v);
            clr[i >> 1] = cur, paint(v, cur ^ 3);
        } else if (!clr[i >> 1]) {
            clr[i >> 1] = cur;
        }
    }
}

inline int count(const int u) {
    int ret = 1; vis[u] = true;
    for (int i = head[u], v; i; i = graph[i].nxt) {
        if (!vis[v = graph[i].to]) {
            ret += count(v);
        }
    }
    return ret;
}

int main() {
    scanf("%d %d", &n, &m);
    rep (i, 1, m) {
        int u, v; scanf("%d %d", &u, &v);
        ++deg[u], ++deg[v], link(u, v);
    }

    rep (u, 1, n) if (!vis[u]) {
        if (deg[u] == 1) paint(u, 1);
        else if (deg[u] > 2) {
            int v = graph[head[u]].to;
            clr[head[u] >> 1] = 1, paint(v, 2);
            if (!vis[u]) paint(u, 2);
        }
    }

    rep (u, 1, n) if (!vis[u] && deg[u] == 2) {
        if (count(u) & 1) return puts("0"), 0;
        for (int v = u, las = 0, cur = 1; ;) {
            for (int i = head[v], w; i; i = graph[i].nxt) {
                if ((w = graph[i].to) != las) {
                    clr[i >> 1] = cur, cur ^= 3, las = v, v = w;
                    break;
                }
            }
            if (v == u) break;
        }
    }

    rep (i, 1, m) printf("%d\n", clr[i]);
    return 0;
}

posted @ 2022-12-23 22:47  Rainybunny  阅读(85)  评论(1编辑  收藏  举报