线性基
线性基这个概念来自于线性代数,即一个矩阵的列向量所能表示空间的一组最小基底。
我们在异或运算下面也可以定义一个空间,即一组数[x1,x2,…,xn][x_{1},x_{2},\ldots,x_{n}][x1,x2,…,xn]由异或运算进行组合,能够表示的空间。
同理,我们定义异或意义下的线性基,表示为是一组数[x1,x2,…,xn][x_{1},x_{2},\ldots,x_{n}][x1,x2,…,xn]的一个基底[b1,b2,…,bm][b_{1},b_{2},\ldots,b_{m}][b1,b2,…,bm],基底的异或空间和原数组的异或空间相同。即如果一个数能由aia_{i}ai表异或示,那么也能用bib_{i}bi异或表示,反之也成立。
还有,线性基是一个极大线性无关子集。
同理与线性代数秩的概念,线性基的大小也是这组数的秩。
构造线性基
下面所有矩阵的概念都是01矩阵。
我们把每一个aia_{i}ai数的二进制看成是矩阵的行向量。进行阶梯化简。
我们定义,bib_{i}bi元素为主元列(最高位111所在的元素位置)为第iii列的行向量。
一开始的矩阵为零矩阵,每次添加一个数xxx,我们从最高位开始扫描xxx,如果遇到了最高位,并且最高位为kkk,首先看他是不是主元,如果bk=0b_{k}=0bk=0,那么说明当前kkk位没有主元,xxx应该充当主元行,因此bk=xb_{k}=xbk=x。如果bk≠0b_{k} \neq 0bk=0,那么说明当前行向量xxx可以被bkb_{k}bk高斯消元,因此令x=x⨁bkx = x \bigoplus b_{k}x=x⨁bk,结果是xxx第kkk位上的111被高斯消元了,继续检索xxx下一个最高位,如果xxx检索到000仍未被插入到bbb中,那么说明最初的xxx可以被当前的线性基表示,因此xxx不是一个无关组向量,直接舍弃。
用代码总结一下这个过程:
#define M64 64
struct LinearBasis
{
ll basis[M64];
LinearBasis()
{
fill(basis, basis + M64, 0);
}
void insert(ll x)
{
for (int i = M64 - 1; i >= 0; i--)
{
if ((x & (1ll << i)) == 0)
continue;
if (basis[i] == 0)
{
basis[i] = x;
break;
}
else
{
x ^= basis[i];
}
}
}
};
合并线性基
我们只需要把一个线性基中的所有向量插入到另外一个线性基上即可。
void merge(const LinearBasis &o)
{
for (int i = 0; i < M64; i++)
insert(o.basis[i]);
}
求秩
在插入的时候,记录秩的大小即可。
#define M64 64
struct LinearBasis
{
ll basis[M64];
int rank;
LinearBasis()
{
fill(basis, basis + M64, 0);
rank = 0;
}
void insert(ll x)
{
for (int i = M64 - 1; i >= 0; i--)
{
if ((x & (1ll << i)) == 0)
continue;
if (basis[i] == 0)
{
basis[i] = x;
rank++;
break;
}
else
{
x ^= basis[i];
}
}
}
void merge(const LinearBasis &o)
{
for (int i = 0; i < M64; i++)
insert(o.basis[i]);
}
};
判断一个数是否是线性基空间中的元素
模拟插入的过程,如果可以插入到bbb中,说明这个数是无关组里的元素,因此不是原线性基空间中的元素,如果不可用插入到bbb中,说明说明这个数不是无关组里的元素,是原线性基空间中的元素。
bool include(ll x)
{
for (int i = M64 - 1; i >= 0; i--)
{
if ((x & (1ll << i)) == 0)
continue;
if (basis[i] == 0)
{
return false;
break;
}
else
{
x ^= basis[i];
}
}
return true;
}
求一个数和线性基能表示的最大元素
设一个数为xxx,我们从最高位开始向后检索,例如检索到第kkk位,如果这个位上是111,那么不管bkb_{k}bk是零还是非零都不可能最优了,因为如果是零,那么不会改变xxx的值,如果是非零,进行异或运算之后第kkk位上的数字反而变成零了。
如果这个位上是000,那么不管bkb_{k}bk是不是零,都是最优的。如果是非零,那么异或之后该位上的数字必定是111,尽管后面的二进制位可能小了,但是不影响最优的结果。
总结一句话,就是取最大值,贪心思想。
ll getMax(ll x = 0)
{
for (int i = 63; i >= 0; i--)
{
x = max(x, x ^ basis[i]);
}
return x;
}
求最小元素
和求最大元素相反,我们也可以讨论四种情况,结果发现,我们只需要贪心最小元素即可。
ll getMin(ll x = 0)
{
for (int i = 63; i >= 0; i--)
{
x = min(x, x ^ basis[i]);
}
return x;
}
求第kkk小的数
我们必须对这个线性基表示成最简阶梯型,之后,设剩下非零的向量的个数为cntcntcnt,那么,这个线性基空间的大小为2cnt2^{cnt}2cnt,即可以表示出2cnt2^{cnt}2cnt个不同的数,因为,对每一个行向量,都可以选择异或或者不异或。
化简之后,最简阶梯型满足进位加法,因此可以根据kkk的二进制位来确定要异或上的元素。
本文探讨了如何在异或运算的基础上定义线性基,包括其性质、如何构造线性基并进行合并,以及如何通过异或操作判断数是否属于该基底空间。介绍了求秩、求最大/最小元素的方法,展示了在信息技术中的实际应用。
558

被折叠的 条评论
为什么被折叠?



