首页 > 其他分享 >选数异或

选数异或

时间:2023-01-09 12:00:10浏览次数:67  
标签:int 选数 枚举 leq 异或 区间 yes

选数异或

给定一个长度为 $n$ 的数列 $A_1,A_2,\ldots,A_n$ 和一个非负整数 $x$,给定 $m$ 次查询,每次询问能否从某个区间 $[l,r]$ 中选择两个下标不同的数使得他们的异或等于 $x$。

输入格式

输入的第一行包含三个整数 $n,m,x$。

第二行包含 $n$ 个整数 $A_1,A_2,\ldots,A_n$。

接下来 $m$ 行,每行包含两个整数 $l_i,r_i$ 表示询问区间 $[l_i,r_i]$。

输出格式

对于每个询问,如果该区间内存在两个数的异或为 $x$ 则输出 yes ,否则输出 no 。

数据范围

对于 $20\%$ 的评测用例,$1 \leq n,m \leq 100$;
对于 $40\%$ 的评测用例,$1 \leq n,m \leq 1000$;
对于所有评测用例,$1 \leq n,m \leq 100000$,$0 \leq x < 2^{20}$,$1 \leq l_i \leq r_i \leq n$,$0 \leq A_i < 2^{20}$。

输入样例:

4 4 1
1 2 3 4
1 4
1 2
2 3
3 3

输出样例:

yes
no
yes
no

样例解释

显然整个数列中只有 $2,3$ 的异或为 $1$。

 

解题思路

  对于一个询问区间$[l, r]$,我们要在这个区间找到两个数使得$a_i \wedge a_j = x$,其中$l \leq i < j \leq r$。因此容易想到可以枚举$j$,然后在前面找到一个满足条件的$i$,即$a_i = a_j \wedge a_j$,因此在枚举的过程中可以开个哈希表来记录枚举过的数,这种做法的时间复杂度为$O(n \cdot m)$。

  或者换一个方法,可以预处理出来每一个数$a_i$左边离它最近的与它匹配的数的下标,如果这个下标在区间$[l, r]$中那么在这个区间中一定存在一组解。定义$f(i)$表示在$a_i$左边与$a_i$配对的最近的一个数的下标,因此如果$a_i$和$a_{f(i)}$是满足条件的一个数对,那么应该有$l \leq f(i) < i \leq r$。因此问题就等价于是否存在一个$i$,$l \leq i \leq r$,使得$l \leq f(i) \leq r$。这种做法的时间复杂度还是$O(n \cdot m)$,因此还需要看看是否存在其他性质。如果在小于$l$的位置中取一个$i$,可以发现这个$i$对应的$f(i)$不可能在区间$[l, r]$范围内,因为$f(i) < i < l$。因此我们可以把$i$的范围扩大到$1 \leq i \leq r$,问题可以变成在$1 \leq i \leq r$是否存在$i$使得$l \leq f(i) \leq r$,进一步,因为$f(i) < i \leq r$,因此就变成能否使得$f(i) \geq l$。因此我们可以求出$f(i)$后(从左往右遍历的过程中开个哈希表记录每个数最新出现的下标),再预处理$g(i) = \max\limits_{1 \leq j \leq i}{f(j)}$,如果有$g(i) \geq l$,那么意味着在以$i$为右端点的前缀中存在一个$f(i) \geq l$。

  AC代码如下:

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 
 4 const int N = 1e5 + 10, M = (1 << 20) + 10;
 5 
 6 int mp[M], g[N];
 7 
 8 int main() {
 9     int n, m, x;
10     scanf("%d %d %d", &n, &m, &x);
11     for (int i = 1; i <= n; i++) {
12         int v;
13         scanf("%d", &v);
14         g[i] = max(g[i - 1], mp[v ^ x]);
15         mp[v] = i;
16     }
17     while (m--) {
18         int l, r;
19         scanf("%d %d", &l, &r);
20         printf("%s\n", g[r] >= l ? "yes" : "no");
21     }
22     
23     return 0;
24 }

  再给出我一开始的做法,比较的麻烦。

  和上一种做法一开始的思路一样,只不过这里是枚举每个满足条件的数对左边的那个数(上一种做法是枚举右边的数)。假设能与$a_i$配对的数是$a_j$,有$j > i$,可以发现当$j$越小,那么$[i, j]$能够被更多的询问区间所完全覆盖,这是因为如果有$a_{j'} = a_{j}$且$j' > j$,并且$[i, j'] \in [l, r]$,那么也一定会有$[i, j] \in [l, r]$,因此区间$[i, j]$的长度越小越好。所以可以一开始把所有的查询区间记录下来(离线处理),然后枚举$i$,找到$a_j$后就有区间$[i, j]$,然后枚举所有的询问区间,把完全覆盖$[i, j]$的都标记成$\text{yes}$,时间复杂度还是$O(n \cdot m)$。可以发现如果对于每个$i$都枚举所有的询问区间,很明显会有冗余,比如对于某个$i$,枚举$l > i$的询问区间是没有意义的,因此我们可以对询问区间按照左端点从小到大排序,每次枚举到到$i$时,把$l \leq i$的询问区间加入到优先队列中,然后这个优先队列是根据区间的右端点来维护一个大根堆,因此每次弹出堆顶元素,如果$r \geq j$,那么这个区间就被标记成$\text{yes}$,同时把这个区间弹出优先队列,时间复杂度是$O(n \log{m})$。

  AC代码如下:

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 
 4 typedef pair<int, int> PII;
 5 
 6 const int N = 1e5 + 10, M = 1 << 20;
 7 
 8 int a[N];
 9 int l[N], r[N];
10 int p[N];
11 vector<int> mp[M];
12 string ans[N];
13 
14 int main() {
15     int n, m, x;
16     scanf("%d %d %d", &n, &m, &x);
17     for (int i = 1; i <= n; i++) {
18         scanf("%d", a + i);
19     }
20     for (int i = 0; i < m; i++) {
21         scanf("%d %d", l + i, r + i);
22         p[i] = i;
23         ans[i] = "no";
24     }
25     sort(p, p + m, [&](int a, int b) {  // 根据询问区间的左端点升序排序
26         return l[a] < l[b];
27     });
28     for (int i = 1; i <= n; i++) {
29         mp[a[i]].push_back(i);  // 哈希表记录每个数所对应的所有下标
30     }
31     priority_queue<PII> pq;
32     for (int i = 1, j = 0; i <= n; i++) {
33         while (j < m && l[p[j]] <= i) { // 把左端点不超过i的区间加入优先队列
34             pq.push({r[p[j]], p[j]});
35             j++;
36         }
37         int t = a[i] ^ x;
38         auto it = upper_bound(mp[t].begin(), mp[t].end(), i);   // 在满足a[i]^a[j]=x的j中找到大于i最小的j
39         if (it != mp[t].end()) {    // 存在与a[i]对应的a[j]
40             while (!pq.empty() && pq.top().first >= *it) {  // 把区间右端点大于等于j的区间标记成yes并弹出队列
41                 ans[pq.top().second] = "yes";
42                 pq.pop();
43             }
44         }
45     }
46     for (int i = 0; i < m; i++) {
47         printf("%s\n", ans[i].c_str());
48     }
49     
50     return 0;
51 }

 

参考资料

  AcWing 4645. 选数异或(寒假每日一题2023):https://www.acwing.com/video/4586/

标签:int,选数,枚举,leq,异或,区间,yes
From: https://www.cnblogs.com/onlyblues/p/17036545.html

相关文章

  • 4645. 选数异或
    4645.选数异或给定一个长度为n的数列A1,A2,⋅⋅⋅,An和一个非负整数x,给定m次查询,每次询问能否从某个区间[l,r]中选择两个下标不同的数使得他们的异或等于x。......
  • 统计异或值在范围内的数对有多少(01字典树)
    1803.统计异或值在范围内的数对有多少-力扣(Leetcode)题意:   思路:很经典的方法,01字典树,同时在树上多维护一个数据,有多少个节点经过当前节点,记为cnt我们可以......
  • P4551 最长异或路径 : 01tire + 树 + 异或
    题P4551最长异或路径https://www.luogu.com.cn/problem/P4551知识背景01tire树,可以用来查找异或的最大值。经典问题如下。在nums中,哪两个数中异或值最大。解决方法:......
  • 关于异或-异或运算及其应用
    概念异或,是一个数学运算符,英文为exclusiveOR,缩写为xor,应用于逻辑运算异或的数学符号为“⊕”,计算机符号为“xor”  如果a、b两个值不相同,则异或结果为1......
  • P1036 [NOIP2002 普及组] 选数(DFS + 不降原则)
    P1036[NOIP2002普及组]选数题意​ 在n个数里选k个数,有多少中选法,使得选出来的数的和为素数。不能重复选。思路​ n很小,直接爆搜,但是如果不使用不降原则的话,就......
  • 子数组的最大异或和问题
    子数组的最大异或和问题作者:Grey原文地址:博客园:子数组的最大异或和问题CSDN:子数组的最大异或和问题题目描述数组中所有数都异或起来的结果,叫做异或和。给定一个数组......
  • 关于异或和一些运算符
    上课的时候经常摸鱼,连异或都没搞懂,今天看map源码的时候才注意到有一个看不懂的计算符staticfinalinthash(Objectkey){inth;return(key==nul......
  • 异或运算及其应用-查找奇数个数的数字
     异或运算功能很强大。用的得当可以提高算法效率。先说一下异或运算的运算法则:      1. a^b=b^a2.a^b^c=a^(b^c)=(a^b)^c  3.......
  • C#实现简单的异或加密,方便处理
    将本地的mp4和ts文件加密为“dj”文件,无法播放。解密则是将“dj”文件解密为mp4或ts文件。usingSystem;usingSystem.Collections.Generic;usingSystem.IO;usingSy......
  • 数组里暴力查找单身狗和'^'异或快速查找单身狗
    intmain(){ chararr[]={1,2,3,4,5,7,5,1,2,3,4}; intsz=sizeof(arr)/sizeof(arr[0]); inti,ret=0; //0^a=a,a^b^a=b,a^a=0,异或满足交换规律,相同为0,反之为1; ......