模板线性基
Posted ty02
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了模板线性基相关的知识,希望对你有一定的参考价值。
线性基
定义:给定数集(S),以异或运算张成的数集与(Span{S})相同的极大线性无关集,称为原数集的一个线性基。
线性基具有以下性质:
- 显然,线性基是原数集的一个子集。
- 线性基的张成集合中一般不包含有数字0。一般给定的数集中不会有0,否则在线性基中加入0即可。
- 张成集合中的每个数都可以唯一表示成基的线性组合。这与第一点是统一的。
- 将线性基中的一个元素通过“倍加变换”(在这里即异或上另一个元素),张成集合不变。
- 线性基中每一个元素的最高位均不同。
由以上性质不难看出,线性基与线性方程组有一些相似之处。事实上,如果把原数集的每一个元素二进制拆分为多项式,把每一项的系数看作一个向量,将所有系数向量看作一个矩阵的行,那么这个构造出的矩阵就可以看作某个线性方程组的系数矩阵。也就是说,对这个矩阵施以异或意义下的初等行变换,即将某些行消成0行,形成的新矩阵与原矩阵行等价。这给了性质5一个很好的阐释:每个元素的最高位对应的是该系数矩阵化为阶梯形后的主元位置,而主元可以将其他行的该项通过行变换消去,结合异或的性质就不难理解了。
以上基于线性代数的讨论表明,可以使用类似高斯消元的行化简算法求得原数集的一个线性基。把每个行向量压成一个数处理,时间复杂度(O(n^2))。
具体来说,对于每一个新加入的数x,我们从高位到低位枚举;如果x的第i位是1(即对应方程左边含有这个元素),并且之前没有出现过该位是1的元素,我们就将x插入线性基(即以这一行作为主元(x_i)的所在行);否则将x异或掉之前出现过的p[i],继续看下一位,直到x被插入或x变为0(即x可以表示为已有基的线性组合)结束。
插入函数代码:
void insert(long long x) {
for (int i = 60; i >= 0; --i)
if ((x >> i) & 1) {
if (!p[i]) {
p[i] = x; break;
}
x ^= p[i];
}
}
luoguP3821
(题解引自洛谷)
先用高斯消元求得一组线性基。由高斯消元可得,该线性基中,主元所在位为1的向量是唯一的。接下来我们可以从高位到低位贪心了。假设现在在考虑第(i)个向量(a_i)(向量降序),其最高位是(x),前(i-1)个向量异或得到的最大整数为(res)。
如果(res)的第xx位是0,则令(res xor a_i),否则不改变(res)。接下来考虑这种做法的正确性。
如果(res)的第(x)位是0但不改变(res),而改变得到(res′)。由于向量降序,比x更高的位和x这一位无法再改变。即使(res)接下来的位全部是1,也没有(x)这一位是1大,即得到(res<res')。因此操作1正确。
如果(res)的第(x)位是1但改变(res),得到(res'),由于向量降序,比x更高的位和x这一位无法再改变。即使(res')接下来的位全部是1,也没有x这一位是1大,即得到(res'<res)。因此操作2正确。
证毕。
代码:
#include <iostream>
#include <cstring>
#include <cstdio>
#include <cctype>
using namespace std;
long long p[70], x;
int n;
void insert(long long x) {
for (int i = 60; i >= 0; --i)
if ((x >> i) & 1) {
if (!p[i]) {
p[i] = x; break;
}
x ^= p[i];
}
}
int main() {
cin >> n;
for (int i = 1; i <= n; ++i) {
cin >> x, insert(x);
}
long long ans = 0;
for (int i = 60; i >= 0; --i)
if ((ans ^ p[i]) > ans) ans ^= p[i];
cout << ans;
return 0;
}
以上是关于模板线性基的主要内容,如果未能解决你的问题,请参考以下文章