D
状压dp, 2000
给出一个 n × m ( n ≤ m ) n\times m(n\leq m) n×m(n≤m) 的 01 01 01矩阵,称任意一个边长为偶数的子正方形中1的个数为奇数的矩阵为好矩阵,求至少改多少个数,可以使其变为好矩阵。如果无论多少次操作都无法变为好矩阵,输出 − 1 -1 −1 。
首先 n ≥ 4 n\geq 4 n≥4 的一定不行,因为 4 个 22 的矩阵个数为奇,则组成的 44 的矩阵个数为偶,于是考虑 n ≤ 3 n\leq 3 n≤3 的。
所以现在只需要解决两个问题:如何判断
n
o
w
now
now 和
p
r
e
pre
pre 之间能够转移,如何求
c
h
a
n
g
e
(
n
o
w
,
i
n
i
t
)
change(now, init)
change(now,init) 。
首先我们这样定义状态:对于
n
=
3
,
s
t
a
t
e
=
a
[
1
]
[
i
]
∗
4
+
a
[
2
]
[
i
]
∗
2
+
a
[
3
]
[
i
]
n=3,state = a[1][i]*4+a[2][i]*2+a[3][i]
n=3,state=a[1][i]∗4+a[2][i]∗2+a[3][i] ,对于
n
=
2
,
s
t
a
t
e
=
a
[
1
]
[
i
]
∗
2
+
a
[
2
]
[
i
]
n=2,state = a[1][i]*2+a[2][i]
n=2,state=a[1][i]∗2+a[2][i] ,第一个问题只需要判断2*2的矩阵中1的数量是否为奇数(若
n
=
3
n=3
n=3 则需要判断2个2*2的矩阵),第二个问题只需要判断
n
o
w
now
now 和初始矩阵有几位不同 。
int n, m;
int a[5][maxn];
int f[maxn][10];
int bit(int x, int y) {
return (x >> y) & 1;
}
int change(int x, int y) { //x 当前状态 y 目标状态
int cnt = 0;
for(int i = 0; i < 4; i++) {
cnt += (bit(x, i) ^ bit(y, i));
}
return cnt;
}
int check2(int pre, int now) {
int cnt = bit(pre, 1) ^ bit(pre, 0) ^ bit(now, 1) ^ bit(now, 0);
return cnt;
}
int check3(int pre, int now) {
int cnt1 = bit(pre, 1) ^ bit(pre, 0) ^ bit(now, 1) ^ bit(now, 0);
int cnt2 = bit(pre, 1) ^ bit(pre, 2) ^ bit(now, 1) ^ bit(now, 2);
return cnt1 && cnt2;
}
void solve() {
memset(f, 0x3f, sizeof f);
cin >> n >> m;
if(n > 3) {
cout << -1 << endl;
return;
}
for(int i = 1; i <= n; i++) {
for(int j = 1; j <= m; j++) {
char c;
cin >> c;
a[i][j] = c - '0';
}
}
if(n == 1) {
cout << 0 << endl;
return;
}
if(n == 2) {
for(int i = 0; i < 4; i++) {
f[1][i] = change(i, a[1][1] * 2 + a[2][1]);
}
for(int j = 2; j <= m; j++) {
for(int now = 0; now < 4; now++) {
for(int pre = 0; pre < 4; pre++) {
if(check2(pre, now)) {
f[j][now] = min(f[j][now],
f[j-1][pre] + change(a[1][j]*2+a[2][j], now));
}
}
}
}
}
if(n == 3) {
for(int i = 0; i < 8; i++) {
f[1][i] = change(i, a[1][1] * 4 + a[2][1] * 2 + a[3][1]);
}
for(int j = 2; j <= m; j++) {
for(int now = 0; now < 8; now++) {
for(int pre = 0; pre < 8; pre++) {
if(check3(now, pre)) {
f[j][now] = min(f[j][now],
f[j-1][pre] + change(now, a[1][j]*4+a[2][j]*2+a[3][j]));
}
}
}
}
}
int ans = INF;
for(int i = 0; i <= 7; i++) chmin(ans, f[m][i]);
cout << ans << endl;
}