CF2147H Maxflow GCD Coloring 题解

Description

给定一个无向图 GG,它有 nn 个顶点,每条边上有一个正整数容量。我们记 maxflow(u,v)\textsf{maxflow}(u,v)

为图中从源点 uu 到汇点 vv 的最大流值。

我们称图 GG好图,如果存在一个整数 d2d \geq 2,使得对于所有不同顶点对 (u,v)(u,v),共有的 n(n1)n \cdot (n-1) 个最大流值 maxflow(u,v)\textsf{maxflow}(u,v) 都能被 dd 整除。特别地,没有边的图显然是好图。

现在给定一张图,你需要给它的顶点染色,使得 每一种颜色对应的顶点诱导出的子图都是好图

请找出所需颜色数目的最小值,并给出一种这样的染色方案。

n50,mn(n1)2n\leq 50,m\leq\frac{n(n-1)}{2}

Solution

先套路性地把最大流转成最小割。然后把初始图就合法的情况判掉。

如果 dd 很大的话我们是难以去刻画所有最大流都是 dd 的倍数的。

所以考虑 d=2d=2 的情况。由于去钦定最小割是偶数是不好做的,因为我们不知道哪些边构成了最小割,那么不妨让所有割集的权值都是偶数。

然后有个结论是每个点连出来的边的边权之和是偶数就能保证所有割集权值都是偶数。证明就考虑把偶边删掉,由于一个割集可以看成给每个点赋一个 0/10/1 的颜色 cic_i,对于一条边 (u,v)(u,v),其在割集等价于 cxcy=1c_x\oplus c_y=1。由于每个点度数都是偶数,所以总割边数中每个 cic_i 都会异或偶数次,也就是 00


现在我们考虑钦定每个导出子图的每个点度数都是偶数,经过打表会发现这个是一定可以在颜色数等于 22 的时候做到的。

具体构造就考虑递归构造,每次先把偶数边去掉,如果不存在奇度点就直接每个点染同样的颜色。否则随便找到一个奇度点 xx,把与其有奇数权值连边的 yy 拿出来,两两加一条权值为 11 的边。

xx 删掉后递归构造,递归回来后把 yy 之间两两连的边去掉。不妨设递归构造出来后之前找到的 yy 中颜色为 00 的集合为 S0S_0,颜色为 11 的集合是 S1S_1

由于 S0+S1|S_0|+|S_1| 是奇数,所以一定存在恰好一个集合目前每个点的度数都是奇数,另一个全是偶数,因为我们刚把它们两两之间的边去掉。xx 的颜色染每个点度数都是奇数的集合对应的颜色,染完后把边加上即可。

对于初始图合法的情况需要用最小割树判断。

时间复杂度:O(n5)O(n^5)

Code

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
#include <bits/stdc++.h>

// #define int int64_t

const int kMaxN = 55, kMaxM = 2.5e3 + 5;

int n, m;
int u[kMaxM], v[kMaxM], w[kMaxM], res[kMaxN][kMaxN];
std::vector<std::pair<int, int>> G[kMaxN];

namespace Dinic {
const int kMaxN = 55, kMaxM = 1e4 + 5;

struct Edge {
int v, w, pre;
} e[kMaxM];

int tot = 1, n, s, t, tail[kMaxN], cur[kMaxN], dep[kMaxN];
bool tag[kMaxN];

void init(int _n, int _s, int _t) {
tot = 1, n = _n, s = _s, t = _t;
for (int i = 1; i <= n; ++i) tail[i] = 0;
}
void adde(int u, int v, int w) { e[++tot] = {v, w, tail[u]}, tail[u] = tot; }
void add(int u, int v, int w) { adde(u, v, w), adde(v, u, 0); }

bool bfs() {
for (int i = 1; i <= n; ++i) cur[i] = tail[i], dep[i] = 1e9;
std::queue<int> q;
dep[s] = 0, q.emplace(s);
for (; !q.empty();) {
int u = q.front(); q.pop();
if (u == t) return 1;
for (int i = tail[u]; i; i = e[i].pre) {
int v = e[i].v, w = e[i].w;
if (w && dep[v] == 1e9) {
dep[v] = dep[u] + 1, q.emplace(v);
}
}
}
return 0;
}

int dfs(int u, int lim) {
if (u == t || !lim) return lim;
int flow = 0;
for (int &i = cur[u]; i; i = e[i].pre) {
int v = e[i].v, w = e[i].w;
if (w && dep[v] == dep[u] + 1) {
int fl = dfs(v, std::min(lim, w));
if (!fl) dep[v] = 1e9;
e[i].w -= fl, e[i ^ 1].w += fl;
lim -= fl, flow += fl;
if (!lim) break;
}
}
return flow;
}

int maxflow() {
int ans = 0;
for (; bfs(); ans += dfs(s, 1e9)) {}
return ans;
}

int calc(int s, int t) {
init(::n, s, t);
for (int i = 1; i <= m; ++i) add(u[i], v[i], w[i]), add(v[i], u[i], w[i]);
return maxflow();
}

void dfs(int u = s) {
if (u == s) std::fill_n(tag + 1, n, 0);
tag[u] = 1;
for (int i = tail[u]; i; i = e[i].pre) {
int v = e[i].v;
if (e[i].w && !tag[v]) dfs(v);
}
}
} // namespace Dinic

void build(std::vector<int> id) {
if (id.size() <= 1) return;
int s = id[0], t = id[1], w = Dinic::calc(s, t);
G[s].emplace_back(t, w), G[t].emplace_back(s, w);
Dinic::dfs();
std::vector<int> idl, idr;
for (auto i : id) (Dinic::tag[i] ? idl : idr).emplace_back(i);
build(idl), build(idr);
}

void dfs(int u, int fa, int *res) {
if (!fa) res[u] = 1e9;
for (auto [v, w] : G[u]) {
if (v == fa) continue;
res[v] = std::min(res[u], w);
dfs(v, u, res);
}
}

void prework() {
for (int i = 1; i <= n; ++i) G[i].clear();
std::vector<int> id;
for (int i = 1; i <= n; ++i) id.emplace_back(i);
build(id);
for (int i = 1; i <= n; ++i) dfs(i, 0, res[i]);
}

bool solve1() {
int d = 0;
for (int i = 1; i <= n; ++i)
for (int j = i + 1; j <= n; ++j)
d = std::__gcd(d, res[i][j]);
if (d == 1) return 0;
std::cout << "1\n" << n << '\n';
for (int i = 1; i <= n; ++i) std::cout << i << " \n"[i == n];
return 1;
}

void solve2() {
static int cnt[kMaxN][kMaxN], deg[kMaxN], col[kMaxN];
static bool del[kMaxN];
for (int i = 1; i <= n; ++i) {
deg[i] = 0;
for (int j = 1; j <= n; ++j) cnt[i][j] = 0;
}
std::function<void(int, int)> add = [&] (int u, int v) {
cnt[u][v] ^= 1, cnt[v][u] ^= 1;
deg[u] ^= 1, deg[v] ^= 1;
};
for (int i = 1; i <= m; ++i) {
if (w[i] & 1) add(u[i], v[i]);
}
for (int i = 1; i <= n; ++i) col[i] = -1, del[i] = 0;
std::function<void()> dfs = [&]() {
int x = 0;
for (int i = 1; i <= n; ++i) {
if (!del[i] && deg[i]) x = i;
}
if (!x) {
for (int i = 1; i <= n; ++i)
if (!del[i])
col[i] = 0;
return;
}
std::vector<int> id;
for (int i = 1; i <= n; ++i)
if (cnt[x][i])
id.emplace_back(i);
del[x] = 1;
for (auto i : id) add(x, i);
for (int i = 0; i < id.size(); ++i)
for (int j = i + 1; j < id.size(); ++j)
add(id[i], id[j]);
dfs();
del[x] = 0;
for (int i = 0; i < id.size(); ++i)
for (int j = i + 1; j < id.size(); ++j)
add(id[i], id[j]);
for (auto i : id) {
int deg = 0;
for (int j = 1; j <= n; ++j)
if (col[j] != -1 && col[i] == col[j])
deg ^= cnt[i][j];
if (deg) add(x, i), col[x] = col[i];
col[x] = col[i] ^ deg ^ 1;
}
};
dfs();
std::vector<int> vec[2];
for (int i = 1; i <= n; ++i) assert(col[i] != -1), vec[col[i]].emplace_back(i);
std::cout << "2\n";
std::cout << vec[0].size() << '\n';
for (auto x : vec[0]) std::cout << x << ' ';
std::cout << '\n';
std::cout << vec[1].size() << '\n';
for (auto x : vec[1]) std::cout << x << ' ';
std::cout << '\n';
}

void dickdreamer() {
std::cin >> n >> m;
for (int i = 1; i <= m; ++i) std::cin >> u[i] >> v[i] >> w[i];
prework();
if (!solve1()) solve2();
}

int32_t main() {
#ifdef ORZXKR
freopen("in.txt", "r", stdin);
freopen("out.txt", "w", stdout);
#endif
std::ios::sync_with_stdio(0), std::cin.tie(0), std::cout.tie(0);
int T = 1;
std::cin >> T;
while (T--) dickdreamer();
// std::cerr << 1.0 * clock() / CLOCKS_PER_SEC << "s\n";
return 0;
}

CF2147H Maxflow GCD Coloring 题解
https://sobaliuziao.github.io/2025/09/22/post/ce040444.html
作者
Egg_laying_master
发布于
2025年9月22日
许可协议