hdu3949
线性基?
线性基在我的理解下是一种向量基底,就是说每个数a[i],都能通过线性基组合出来。又因为线性基只有01,所以系数只有01
这道题感觉就是确定每一个二进制位能不能通过xor组合出来,然后因为二进制可以通过贪心确定大小,因为选择最高位肯定是最大的,所以我们可以拆分k来确定每一位选还是不选。
消元之后就能确定每个二进制位是否能选出来,因为高斯消元完了消出一组解,就肯定能通过选或不选某个数组合出来,肯定不会矛盾,比如说两个位是1,他们的组合矛盾了,这是不可能的,因为高斯消元求出了一组方程的解,解肯定满足任意一条方程。
now的意思是能够通过now个元素组合出所有的数,这样的意思也就是有2^now-1个元素,其他的数都是重复。
又理解了一下,就是把每个数消成只有一个1或0个1,0个1选不选无所谓,1个1有选或不选。所以就是选或不选变成了两个数,也就是我们有多少个消完后的a[i]不是0,那么我们就有2^个数-1个不重复的数(0不能出现) 这个东西和向量挺像的。
上面是错的。。。不是最多有一个1,可以有很多个1,只有两个数3和4就不是只有一个1了。基底可以表示集合中所有的数。。。每个基底都是由几个数合成的,如果我们想用几个基底合成新的数的时候,如果我们缺少基底的元素,那么我们就加上,如果需要用以前用过的数,那么直接消掉,因为异或和自己异或可以消掉。这样总能组成一个新的数。线性基就是帮助我们找了一个最简的集合,里面的元素选或不选就能构造出新的数,这也就是线性基的意义所在,对于线性基的元素只用考虑选或不选。线性基的用处似乎还有很多,最重要的就是可以选择异或第k大的异或和,能很快地查询一个异或和的信息。
还有一点不明白...
#include<bits/stdc++.h> using namespace std; typedef long long ll; const int N = 10010; int n, m, now; ll a[N], bin[70]; void gauss() { now = 1; for(int i = 61; i >= 0; --i) { int j = now; while(j <= n && !(a[j] & bin[i])) ++j; if(j > n) continue; //这位没有,不能弄出来0 swap(a[now], a[j]); for(int j = 1; j <= n; ++j) if(j != now && a[j] & bin[i]) a[j] ^= a[now]; ++now; //下一位 } --now; //可以弄出来2^now-1个数 } ll query(ll k) { ll ret = 0; if(now != n) --k; if(k >= bin[now]) return -1; for(int i = 1; i <= now; ++i) if(k & bin[now - i]) ret ^= a[i]; return ret; } int main() { bin[0] = 1; for(int i = 1; i <= 61 ; ++i) bin[i] = bin[i - 1] << 1; int T, cas = 0; scanf("%d", &T); while(T--) { printf("Case #%d:\n", ++cas); scanf("%d", &n); for(int i = 1; i <= n; ++i) scanf("%lld", &a[i]); gauss(); scanf("%d", &m); while(m--) { ll k; scanf("%lld", &k); printf("%lld\n", query(k)); } } return 0; }