Live2D

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

Description

  Link.

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

  n,m105.

Solution

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

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

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

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

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

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

  其实所有 deg3 点都可以.

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

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 @   Rainybunny  阅读(87)  评论(1编辑  收藏  举报
相关博文:
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示