蒟蒻的我显然不知道什么是"基",更不知道一大堆概念等乱七八糟的东西,就通俗的讲好了。
OI中线性基的用处 :一个序列a1,a2....an,我们选出任意个数使得答案的异或值最大。
线性基的本质就是找出一些数,使得这些数对答案的贡献与全局的贡献相同(值域相同)。
这道题我们考虑构造法按位贪心。对于每个数,我们从高位往低位扫,若这个数对这一位有贡献(二进制下为1),我们就把这个加入集合,停止扫描(一个数显然不会被选两次啦)。
查询时就从高位到低位扫集合的数组,若使答案变大,则贪心异或。
因为集合中只有log个数,所以时间复杂度为O(nlogn),比高斯消元快许多。
找集合
1 for(intt i=1;i<=n;i++)//扫描每个数字 2 { 3 for(intt j=51;j>=0;j--)//按位开桶(集合) 4 { 5 if(!(a[i]>>j))//这个数对这一位没有贡献,看下一位 6 continue; 7 if(!p[j])//这一位还没有数 8 { 9 p[j]=a[i];//赋值 10 break; 11 } 12 a[i]^=p[j];//这里很关键:我们要保证选到的数对集合中的这一位有贡献,贪心才能够成立。 13 } 14 }
查询
for(intt i=51;i>=0;i--) if((ans^p[i])>ans) ans^=p[i];//贪心
全代码
#include<cstdio> #include<algorithm> #include<cstring> using namespace std; #define intt long long #define N 500 intt a[N]; intt p[N]; intt ans=0; intt n; int main() { scanf("%lld",&n); for(intt i=1;i<=n;i++) scanf("%lld",&a[i]); for(intt i=1;i<=n;i++)//扫描每个数字 { for(intt j=51;j>=0;j--)//按位开桶(集合) { if(!(a[i]>>j))//这个数对这一位没有贡献,看下一位 continue; if(!p[j])//这一位还没有数 { p[j]=a[i];//赋值 break; } a[i]^=p[j];//这里很关键:我们要保证选到的数对集合中的这一位有贡献,贪心才能够成立。 } } for(intt i=51;i>=0;i--) if((ans^p[i])>ans) ans^=p[i];//贪心 printf("%lld",ans); return 0; }
线性基还有许多应用,如果哪里没懂请在讨论中问我,我会解答。