题解 CF704C 【Black Widow】
CF704C Black Widow
题意
- 有
m 个布尔变量x_{1\dots m} ,设x_{-i}=\neg x_{i} 。 - 给定
n 个形如x_i 或x_i \operatorname{or} x_j 的表达式,保证x_i 和x_{-i} 在所有表达式中一共只会出现最多两次。 - 询问在
2^m 种不同的取值方案中,有多少种方案使得所有表达式的值的异或为1 。 -
题解
将每个表达式看成一个点。定义一个点的值为这个点对应的表达式的值,一个连通块的值为连通块内点的值的异或,整个图的值为所有连通块的值的异或。我们要求的就是整个图的值为
由于保证
假设我们求出来了每个连通块的值为
设
c_{i,0/1} 表示第i 个连通块值为0/1 的方案数。设
f_{i,0/1} 表示只考虑前i 个连通块,值为0/1 的方案数。初始状态:
f_{0,0} = 1 。目标状态:
f_{*,1} 。转移:
f_{i,j} = f_{i-1,j} \times c_{i,0} + f_{i-1,j \operatorname{xor} 1} \times c_{i,1} 。
现在考虑如何对于每个连通块求
注意到连出来的图一定由若干个连通块组成的,而连通块只有四种形态,我们分别考虑。
一个点的链
表达式形如
表达式形如
一个点的环
表达式形如
表达式形如
至少两个点的链
这种形态下,链的两端对应的表达式可能形如
如果形如
设第
i 个点上的表达式为x_{p_i} \operatorname{or} x_{q_i} ,其中x_{|q_{i-1}|} = x_{|p_i|} 。设
g_{i,0/1,0/1} 表示只考虑前i 个点,值为0/1 ,x_{|q_i|} = 0/1 的方案数。初始状态:对于
x_{|p_1|} 的每种取值,g_{0,0,x_{|p_1|}} = 1 。目标状态:对于
x_{|q_{*}|} 的每种取值,c_0 \leftarrow g_{*,0,x_{|q_{*}|}} ,c_1 \leftarrow g_{*,1,x_{|q_{*}|}} 。转移:对于
x_{|p_i|},x_{|q_i|} 的每种取值,g_{i,(x_{p_i} \operatorname{or} x_{q_i}) \operatorname{xor} j,x_{|q_i|}} \leftarrow \sum_{j=0}^1 g_{i-1,j,x_{|p_i|}} 。
至少两个点的环
随便选一条边断开,转化成至少两个点的链的形态。
对于断开处的变量,枚举它的值
然后用至少两个点的链的方法做即可。
代码
const int N = 1e5 + 7;
int n, m, d[N], k, v[N];
vi a[N], b[N], e[N], s;
modint f[N][2], c[2], g[N][2][2], ans = 1;
void dfs(int x) {
v[x] = k, s.pb(x);
for (int y : e[x])
if (!v[y]) dfs(y);
}
inline void dp(int l1, int r1, int l2, int r2) {
int t = s.size();
for (int i = 0; i <= t; i++)
for (int j = 0; j < 2; j++)
for (int k = 0; k < 2; k++)
g[i][j][k] = 0;
for (int i = l1; i <= r1; i++) g[0][0][i] = 1;
if (!a[s[0]][1] || (abs(a[s[0]][1]) != abs(a[s[1]][0]) && abs(a[s[0]][1]) != abs(a[s[1]][1]))) swap(a[s[0]][0], a[s[0]][1]);
for (int i = 1; i < t; i++)
if (abs(a[s[i]][0]) != abs(a[s[i-1]][1]))
swap(a[s[i]][0], a[s[i]][1]);
for (int i = 1; i <= t; i++) {
int x = s[i-1];
for (int p = 0; p < 2; p++)
for (int q = 0; q < 2; q++)
for (int j = 0; j < 2; j++)
g[i][(((a[x][0]<0)^p)|((a[x][1]<0)^q))^j][q] += g[i-1][j][p];
}
for (int i = 0; i < 2; i++)
for (int j = l2; j <= r2; j++)
c[i] += g[t][i][j];
}
inline void solve(int o) {
if (s.size() == 1u) {
int x = s[0];
if (a[x].size() == 1u) return c[0] = c[1] = 1, void();
if (abs(a[x][0]) != abs(a[x][1])) return c[0] = 1, c[1] = 3, void();
if (a[x][0] == a[x][1]) return c[0] = c[1] = 1, void();
return c[0] = 0, c[1] = 2, void();
}
int rt = 0;
for (int x : s)
if (d[x] == 1) rt = x;
c[0] = c[1] = 0;
if (rt) {
for (int x : s) v[x] = 0;
s.clear(), dfs(rt);
int x = s[0], l1 = 0, r1 = 1, l2 = 0, r2 = 1;
if (a[x].size() == 1u) r1 = 0, a[x].pb(0);
x = s.back();
if (a[x].size() == 1u) r2 = 0, a[x].pb(0);
return dp(l1, r1, l2, r2);
}
dp(0, 0, 0, 0), dp(1, 1, 1, 1);
}
int main() {
rd(n), rd(m);
for (int i = 1, o, x; i <= n; i++) {
rd(o);
while (o--) rd(x), a[i].pb(x), b[abs(x)].pb(i);
}
for (int i = 1; i <= m; i++)
if (b[i].size() == 2u) {
int x = b[i][0], y = b[i][1];
e[x].pb(y), e[y].pb(x), ++d[x], ++d[y];
} else if (!b[i].size()) ans *= 2;
f[0][0] = 1;
for (int i = 1; i <= n; i++)
if (!v[i]) {
s.clear(), ++k, dfs(i), solve(k);
for (int j = 0; j < 2; j++)
f[k][j] = f[k-1][j] * c[0] + f[k-1][j^1] * c[1];
}
ans *= f[k][1];
print(ans);
return 0;
}