题意

定义 NAND(与非)运算,其运算结果为真当且仅当两个输入的布尔值不全为真,也就是 A NAND B = NOT(A AND B) ,运算位数不会超过 \(k\) 位,

给你 \(n\) 个整数 \(A_i\) ,这些数能任意进行无数次与非运算,最后问能运算出多少个在 \([L, R]\) 区间的数。

\(k \le 60, n \le 1000, A_i < 2^k, 0 \le L \le R \le 10^{18}\)

题解

参考了 kczno1 孔爷的题解。

这个运算初看不太优美,其实我们可以利用它的一些性质。

由于 A NAND A = NOT (A AND A) = NOT A 所以我们就可以得到了 NOT (非) 运算。

进一步我们利用 NOTNOT(A NAND B) = NOT(NOT(A AND B)) = A AND B ,就可以得到了 AND (与)运算。

这样这个运算就变得十分优秀了,我们就转化成进行 NOTAND 然后我们就可以得到 所有二进制操作 了。

然后利用这个性质,我们就可以得到一个更加有用的性质。

对于第 \(i\) 位和第 \(j\) 位,如果所有 \(A_k\) 的第 \(i\) 位和第 \(j\) 位相同,那么最后的结果对于 \(i, j\) 这两位一定是一样的。

否则这两位的取值互不影响,可以任意取都能构造出一组合法方案。

我们就能把 \(k\) 位数划分成许多个等价类,每个等价类里面的元素取值都必须一样。

然后为了算 \([l, r]\) 区间的答案,我们令 \(Calc(r)\) 为 \(\le r\) 能凑出来的数。

那么答案为 \(Calc(r) - Calc(l - 1)\) 。至于如何算 \(Calc(x)\) 呢?我们按位考虑就行了。

  1. 具体的,如果枚举的位为 \(0\) ,那么忽略。

  2. 如果为 \(1\) ,假设这一位不能选,那么接下来以任意选而不会超也不会重复,所以方案数加上 \(2^{sum}\) ( \(sum\) 为接下来的集合的个数) 然后退出。

    如果能选。要么不选,那么 \(2^{sum-1}\) ;要么将集合中的数全部选了,再接着枚举后面。、

复杂度是 \(O(nk)\) 的,不知道为什么 \(n\) 只开 \(1000\) 。。。石乐志。

代码

#include <bits/stdc++.h>

#define For(i, l, r) for (register int i = (l), i##end = (int)(r); i <= i##end; ++i)
#define Fordown(i, r, l) for (register int i = (r), i##end = (int)(l); i >= i##end; --i)
#define Rep(i, r) for (register int i = (0), i##end = (int)(r); i < i##end; ++i)
#define Set(a, v) memset(a, v, sizeof(a))
#define Cpy(a, b) memcpy(a, b, sizeof(a))
#define debug(x) cout << #x << ": " << (x) << endl using namespace std; typedef long long ll; template<typename T> inline bool chkmin(T &a, T b) { return b < a ? a = b, 1 : 0; }
template<typename T> inline bool chkmax(T &a, T b) { return b > a ? a = b, 1 : 0; } template<typename T>
inline T read() {
T x(0), sgn(1); char ch(getchar());
for (; !isdigit(ch); ch = getchar()) if (ch == '-') sgn = -1;
for (; isdigit(ch); ch = getchar()) x = (x * 10) + (ch ^ 48);
return x * sgn;
} void File() {
#ifdef zjp_shadow
freopen ("2728.in", "r", stdin);
freopen ("2728.out", "w", stdout);
#endif
} typedef long long ll; const int N = 1e3 + 1e2, K = 60; ll a[N], sta[N], now, full; int n, k, sum[K]; inline ll Calc(ll x) {
if (x >= full) return 1ll << sum[k - 1];
ll res = 0;
for (int i = k - 1; x >= 0 && i >= 0; -- i)
if (x >> i & 1) {
if (sta[i]) {
res += 1ll << (sum[i] - 1); x -= sta[i];
}
else {
res += 1ll << sum[i]; break;
}
}
return res + (x == 0);
} int main () { File(); n = read<int>();
full = (1ll << (k = read<int>())) - 1; ll l = read<ll>(), r = read<ll>();
For (i, 1, n) a[i] = read<ll>(); ll have = 0;
Fordown (i, k - 1, 0) if (!(have >> i & 1)) {
ll now = full;
For (j, 1, n)
now &= (a[j] >> i & 1) ? a[j] : ~a[j];
have |= (sta[i] = now); sum[i] = 1;
}
For (i, 1, k - 1) sum[i] += sum[i - 1]; printf ("%lld\n", Calc(r) - Calc(l - 1)); return 0; }

BZOJ 2728: [HNOI2012]与非(位运算)的更多相关文章

  1. BZOJ 2728: [HNOI2012]与非

    2728: [HNOI2012]与非 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 786  Solved: 371[Submit][Status][ ...

  2. BZOJ 2728 HNOI2012 与非 高斯消元

    题目大意:给定k位二进制下的n个数,求[l,r]区间内有多少个数能通过这几个数与非得到 首先观察真值表 我们有A nand A = not A 然后就有not ( A nand B ) = A and ...

  3. 【BZOJ 2728】 2728: [HNOI2012]与非 (线性基?)

    2728: [HNOI2012]与非 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 813  Solved: 389 Description Inpu ...

  4. BZOJ 4245: [ONTAK2015]OR-XOR 贪心 + 位运算

    Description 给定一个长度为n的序列a[1],a[2],...,a[n],请将它划分为m段连续的区间,设第i段的费用c[i]为该段内所有数字的异或和,则总费用为c[1] or c[2] or ...

  5. HYSBZ(BZOJ) 4300 绝世好题(位运算,递推)

    HYSBZ(BZOJ) 4300 绝世好题(位运算,递推) Description 给定一个长度为n的数列ai,求ai的子序列bi的最长长度,满足bi&bi-1!=0(2<=i<= ...

  6. Java 位运算(移位、位与、或、异或、非)

    Java提供的位运算符有:左移( << ).右移( >> ) .无符号右移( >>> ) .位与( & ) .位或( | ).位非( ~ ).位异或( ...

  7. BZOJ 1087 互不侵犯King (位运算)

    题解:首先,这道题可以用位运算来表示每一行的状态,同八皇后的搜索方法,然后对于限制条件不相互攻击,则只需将新加入的一行左右移动与上一行相&,若是0则互不攻击,方案可行.对于每种方案,则用递推来 ...

  8. 【Java基础】3、Java 位运算(移位、位与、或、异或、非)

    public class Test { public static void main(String[] args) { // 1.左移( << ) // 0000 0000 0000 0 ...

  9. 【BZOJ2728】[HNOI2012]与非 并查集+数位DP

    [BZOJ2728][HNOI2012]与非 Description Input 输入文件第一行是用空格隔开的四个正整数N,K,L和R,接下来的一行是N个非负整数A1,A2……AN,其含义如上所述.  ...

随机推荐

  1. python的四种内置数据结构

    对于每种编程语言一般都会规定一些容器来保存某些数据,就像java的集合和数组一样python也同样有这样的结构 而对于python他有四个这样的内置容器来存储数据,他们都是python语言的一部分可以 ...

  2. Java向下转型的意义

    一开始学习 Java 时不重视向下转型.一直搞不清楚向下转型的意义和用途,不清楚其实就是不会,那开发的过程肯定也想不到用向下转型. 其实向上转型和向下转型都是很重要的,可能我们平时见向上转型多一点,向 ...

  3. Ubuntu 12.04 安装socks5代理服务器dante-server

    dante-server是一个很好的socks4/5代理服务器软件. 使用apt-get安装   1 apt-getinstall dante-server 添加一个用户   1 2 useradd ...

  4. Notepad++快捷使用

    用Notepad++写代码,要是有一些重复的代码想copy一下有木有简单的方法呢,确实还是有的不过也不算太好用.主要是应用键盘上的 Home 键 和 End 键.鼠标光标停留在一行的某处,按 Home ...

  5. Oracle Profile 配置文件

    Profile是用户的配置文件,它是密码限制,资源限制的命名集合.利用profile 可以对数据库用户进行基本的资源管理,密码管理. 1 创建profile 的语法 create profile pr ...

  6. js怎么能取得多选下拉框选中的多个值?

    方法:获取多选下拉框对象数组→循环判断option选项的selected属性(true为选中,false为未选中)→使用value属性取出选中项的值.实例演示如下: 1.HTML结构 1 2 3 4 ...

  7. js的日期操作:String转date日期格式、求日期差

    一.在js中String类型转成date格式 var date = new Date("2018-9-21 14:58:43");//就是这么简单 二.date转String类型就 ...

  8. __new__和__init__的区别

    __new__是一个静态方法,而__init__是一个实例方法. __new__方法会返回一个创建的实例,而__init__什么都不返回. 只有在__new__返回一个cls的实例时后面的__init ...

  9. 剑指offer(1)

    题目: 在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数. ...

  10. SpringBoot 4.SpringBoot 整合 devtools 实现热部署

    一.添加 devtools 依赖 <!-- Spring boot 热部署 : 此热部署会遇到 java.lang.ClassCastException 异常 --> <!-- op ...