[HNOI 2018]寻宝游戏
Description
给出 $n$ 个 $m$ 位的二进制数,在每一个二进制数间插入一个 & 或 | ,第 $0$ 个数为 $0$ , $0,1$ 间也要插入符号,共插入 $n$ 个符号。
给出 $q$ 组询问,每组询问也给出一个二进制数,询问有多少种方式使得从左至右运算后结果为该数。
$1\leq n\leq 1000,1\leq m\leq 5000,1\leq q\leq 1000$
Solution
人类智慧题。
我们可以把每一位单独拿出来处理,将每一位压成一个 $n$ 位的二进制数,第 $i$ 位为 $a_i$ 。越靠右位数越高。
我们对于生成的符号序列,将 & 记为 $1$ , 将 | 记为 $0$ 。同样越靠右位数越高。记为 $x$ 。
考验人类智慧的就是,有这样一个结论:若第 $i$ 位运算结果为 $1$ ,当且仅当 $x<a_i$ 。这个可以感性理解一下,比较直观。
那么我们考虑将 $a_i$ 从大到小排序,我们只需找到这样的唯一一个断点,满足断点以左的 $a_i$ 对应的那一位均为 $1$ ,断点以右的均为 $0$ 。
假设断点右边一位为 $loc$ ,那么答案就是 $a_{loc-1}-a_{loc}$ 。如果没有断点,显然无解。
Code
#include <bits/stdc++.h>
using namespace std;
const int N = 1000+5, M = 5000+5, yzh = 1000000007;
int n, m, q, mp[M], cg[M];
char ch[M];
struct tt {
int b[N], id;
bool operator < (const tt &c) const {
for (int i = n; i >= 1; i--)
if (b[i] != c.b[i]) return b[i] > c.b[i];
}
}a[M];
int get_ans(int o) {
int x = 0, y = 0;
for (int i = n; i >= 1; i--) x = (2ll*x%yzh+a[o-1].b[i])%yzh;
for (int i = n; i >= 1; i--) y = (2ll*y%yzh+a[o].b[i])%yzh;
return ((x-y+(o == 1))%yzh+yzh)%yzh;
}
void work() {
scanf("%d%d%d", &n, &m, &q);
for (int i = 1; i <= n; i++) {
scanf("%s", ch+1);
for (int j = 1; j <= m; j++) a[j].b[i] = ch[j]-'0';
a[0].b[i] = 1;
}
for (int i = 1; i <= m; i++) a[i].id = i;
sort(a+1, a+m+1);
for (int i = 1; i <= m; i++) mp[a[i].id] = i;
while (q--) {
scanf("%s", ch+1);
for (int i = 1; i <= m; i++) cg[mp[i]] = ch[i]-'0';
int flag = 0, t = 0;
for (int i = 1; i <= m; i++) {
if (cg[i] && t) {flag = 1; break; }
if (cg[i] == 0) t = 1;
}
if (flag) puts("0");
else {
int flag = 0;
for (int i = 1; i <= m; i++)
if (cg[i] == 0) {printf("%d\n", get_ans(i)); flag = 1; break; }
if (!flag) printf("%d\n", get_ans(m+1));
}
}
}
int main() {work(); return 0; }
博主蒟蒻,随意转载。但必须附上原文链接:http://www.cnblogs.com/NaVi-Awson/,否则你会终生找不到妹子!!!

浙公网安备 33010602011771号