【线性基】51nod1312 最大异或和&LOJ114 k大异或和
1312 最大异或和
- 第一行一个整数N,且1<=N<=50
- 接下来N行每行一个整数S[i],且0<=S[i]<=1,000,000,000,000,000 (10^15)
- 一个整数,即最后集合可能的最大值SUM。
- 3
- 1
- 2
- 3
- 8
题解
处理出来线性基直接求得结果即可
代码
- //by 减维
- #include<set>
- #include<map>
- #include<ctime>
- #include<cmath>
- #include<queue>
- #include<bitset>
- #include<vector>
- #include<cstdio>
- #include<cstring>
- #include<iostream>
- #include<algorithm>
- #define ll long long
- #define il inline
- #define db double
- #define rg register
- #define mpr make_pair
- #define maxn 105
- #define eps 1e-8
- #define inf (1<<30)
- #define pi 3.1415926535897932384626L
- using namespace std;
- inline int read()
- {
- int ret=;bool fla=;char ch=getchar();
- while((ch<''||ch>'')&&ch!='-')ch=getchar();
- if(ch=='-'){fla=;ch=getchar();}
- while(ch>=''&&ch<=''){ret=ret*+ch-'';ch=getchar();}
- return fla?-ret:ret;
- }
- int n,cnt;
- ll mx,ans,a[maxn],b[maxn],p[],bin[];
- int main()
- {
- n=read();
- bin[]=;for(int i=;i<=;++i) bin[i]=bin[i-]<<;
- for(int i=;i<=n;++i) scanf("%lld",&a[i]),b[i]=a[i];
- for(int i=;i<=n;++i)
- for(int j=;j>=;--j)
- if(a[i]&bin[j])
- if(!p[j]){p[j]=a[i];break;}
- else a[i]^=p[j];
- for(int i=;i>=;--i)
- for(int j=i-;j>=;--j)
- if(p[i]&bin[j]) p[i]^=p[j];
- for(int i=;i>=;--i)
- if(p[i])
- {
- if((mx^p[i])>mx) mx^=p[i];
- cnt++;
- }
- ans+=mx*(n-cnt+);
- cnt--;
- for(int i=;cnt&&i<=;i++)
- if(p[i]) ans+=(mx^p[i]),cnt--;
- printf("%lld",ans);
- return ;
- }
#114. k 大异或和
题目描述
这是一道模板题。
给由 n n n 个数组成的一个可重集 S S S,每次给定一个数 k k k,求一个集合 T⊆S T \subseteq S T⊆S,使得集合 T T T 在 S S S 的所有非空子集的不同的异或和中,其异或和 T1xorT2xor…xorT|T| 是第 k k k 小的。
输入格式
第一行一个数 n n n。
第二行 n n n 个数,表示集合 S S S。
第三行一个数 m m m,表示询问次数。
第四行 m m m 个数,表示每一次询问的 k k k。
输出格式
输出 m m m 行,对应每一次询问的答案,第 k k k 小的异或和。如果集合 S S S 的所有非空子集中,不同的异或和数量不足 k k k,输出 −1 -1 −1。
样例
样例输入
3
1 2 3
5
1 2 3 4 5
样例输出
0
1
2
3
-1
数据范围与提示
1≤n,m≤10^5,0≤Si≤250,0≤Si≤250
题解
要把线性基求出来后再消下元
对于询问,把k二进制拆分再求即可
代码
- //by 减维
- #include<set>
- #include<map>
- #include<ctime>
- #include<cmath>
- #include<queue>
- #include<bitset>
- #include<vector>
- #include<cstdio>
- #include<cstring>
- #include<iostream>
- #include<algorithm>
- #define ll long long
- #define il inline
- #define db double
- #define rg register
- #define mpr make_pair
- #define maxn 100005
- #define eps 1e-8
- #define inf (1<<30)
- #define pi 3.1415926535897932384626L
- using namespace std;
- inline int read()
- {
- int ret=;bool fla=;char ch=getchar();
- while((ch<''||ch>'')&&ch!='-')ch=getchar();
- if(ch=='-'){fla=;ch=getchar();}
- while(ch>=''&&ch<=''){ret=ret*+ch-'';ch=getchar();}
- return fla?-ret:ret;
- }
- int n,m,cnt;
- ll p[],bin[],a[];
- int main()
- {
- n=read();int pd=;
- bin[]=;for(int i=;i<=;++i) bin[i]=bin[i-]<<;
- for(int i=;i<=n;++i)
- {
- ll x;scanf("%lld",&x);
- for(int j=;j>=;--j)
- if(x&bin[j]){
- if(!p[j]){p[j]=x;break;}
- x^=p[j];
- }
- if(!x) pd=;
- }
- for(int i=;i>=;--i)
- for(int j=i-;j>=;--j)
- if(bin[j]&p[i]) p[i]^=p[j];
- for(int i=;i<=;++i) if(p[i]) a[cnt++]=p[i];
- m=read();
- ll k;
- for(int i=;i<=m;++i)
- {
- scanf("%lld",&k);k-=pd;
- ll ans=;
- if(k>bin[cnt]-){puts("-1");continue;}
- for(int j=cnt-;j>=;--j)
- if(k&bin[j]) ans^=a[j];
- printf("%lld\n",ans);
- }
- return ;
- }
【线性基】51nod1312 最大异或和&LOJ114 k大异或和的更多相关文章
- LOJ114 k大异或和
传送门 (vjudge和hdu也有但是我觉得LOJ好看!而且限制少!) 不过本题描述有误,应该是k小. 首先我们需要对线性基进行改造.需要把每一位改造成为,包含最高位的能异或出来的最小的数. 为啥呢? ...
- LOJ.114.K大异或和(线性基)
题目链接 如何求线性基中第K小的异或和?好像不太好做. 如果我们在线性基内部Xor一下,使得从高到低位枚举时,选base[i]一定比不选base[i]大(存在base[i]). 这可以重构一下线性基, ...
- Loj 114 k大异或和
Loj 114 k大异或和 构造线性基时有所变化.试图构造一个线性基,使得从高到低位走,异或上一个非 \(0\) 的数,总能变大. 构造时让任意两个 \(bas\) 上有值的 \(i,j\) ,满足 ...
- [LOJ#114]k 大异或和
[LOJ#114]k 大异或和 试题描述 这是一道模板题. 给由 n 个数组成的一个可重集 S,每次给定一个数 k,求一个集合 T⊆S,使得集合 T 在 S 的所有非空子集的不同的异或和中,其异或和 ...
- LibreOJ #114. k 大异或和
二次联通门 : LibreOJ #114. k 大异或和 /* LibreOJ #114. k 大异或和 WA了很多遍 为什么呢... 一开始读入原数的时候写的是for(;N--;) 而重新构造线性基 ...
- LOJ114 k大(xiao)异或和(线性基)
构造线性基后将其消至对任意位至多只有一个元素该位为1.于是就可以贪心了,将k拆成二进制就好.注意check一下是否能异或出0. #include<iostream> #include< ...
- 【loj114】k大异或和 线性基+特判
题目描述 给由 $n$ 个数组成的一个可重集 $S$ ,每次给定一个数 $k$ ,求一个集合 $T⊆S$ ,使得集合 $T$ 在 $S$ 的所有非空子集的不同的异或和中,其异或和 $T_1 ...
- 第k大异或值
这道题与2018年十二省联考中的异或粽子很相像,可以算作一个简易版: 因为这不需要可持久化: 也就是说求任意两个数异或起来的第k大值: 首先把所有数放进trie里. 然后二分答案,枚举每个数,相应地在 ...
- hdu 3949 第k大异或组合
题意: 给你一些数,其中任选一些数(大于等于一个),那么他们有一个异或和. 求所有这样的异或和的第k小. 我们可以将每一位看成一维,然后就是给我们n个60维的向量,求它们线性组合后得到的向量空间中,第 ...
随机推荐
- 运行自己的shell脚本
shell脚本可以直接./**.sh,也可以bash **.sh 我用./**.sh运行自己写的一个脚本,会出现如下的错误: bnrc@bnrc:~$ ./pixel.sh bash: ./pixel ...
- Python实现注册和登录
一.注册账号需要实现的功能 1.输入:用户名,密码,密码确认 2.限制1:输入的账号和密码不能为空 3.限制2:两次输入密码必须一致 4.限制3:用户名不能重复 5.限制4:错误次数为4次 6.用字典 ...
- bootstrap table 自定义checkbox样式
//css <style> .checkbox-custom { position: relative; padding: 0 15px 0 25px; margin-bottom: 7p ...
- dom事件操作例题,电子时钟,验证码,随机事件
dom事件操作 当事件发生时,可以执行js 例子: 当用户点击时,会改变<h1>的内容: <h1 onClick="this.innerHTML='文本更换'"& ...
- Oracle 换行符 空格符 回车符
① 换行符 chr(10)② 回车符 chr(13) ③ 空格符 chr(9) 例1:效果对比.chr(10)在一个字段中换行显示一列数据,chr(13)同样是换行显示一行数据,chr(9)会显示一个 ...
- Linux 命令大全提供 500 多个 Linux 命令搜索
Linux Command 在这里维持一个持续更新的地方 516 个 Linux 命令大全,内容包含 Linux 命令手册.详解.学习,值得收藏的 Linux 命令速查手册.请原谅我写了个爬虫,爬了他 ...
- jquery简易的三级导航
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> ...
- js时间转换
1. 将时间戳转换成日期格式 // 简单的一句代码 var date = new Date(时间戳); //获取一个时间对象 /** 1. 下面是获取时间日期的方法,需要什么样的格式自己拼接起来就好了 ...
- 20190103(GIL,池,阻塞,同步异步)
GIL锁 什么是GIL GIL全局解释器锁,是防止多个线程在同一时间同时执行的.CPython解释器特有的一种互斥锁. 每一个py文件都会有自己的解释器,也就是说不同py文件的GIL都是独立的, ps ...
- day13内置函数
内置函数 一.三元表达式 def max2(x,y): if x>y: return x else: return y res=max2(10,11) print(res) 三元表达式仅应用于: ...