【题意分析】
给你一个可重复数集,要求从中选取一个关于异或空间线性无关的子集,使子集的权值和最大。
【解题思路】
定义:一个有序对(S,I)称为拟阵当且仅当该有序对满足以下性质:
1.有穷性:S是一个有限集。
2.遗传性:I是S的一类具有遗传性质的非空子集族。具体地说,∀B∈I,若A⊂B,则A∈I。
3.交换性:I满足交换性。具体地说,∀A,B∈I不妨设|A|<|B|,必定存在某一元素x∈B-A,使A∪{x}∈I。
衍生概念:
独立子集:给定拟阵M=(S,I),A称为S的独立子集当且仅当A∈I。
可扩展元素:给定拟阵M=(S,I)与独立子集A∈I,若存在x∈S但x∉A,使得A∪{x}∈I,则称x为A的可扩展元素。
我们可以构造关于可重数集S的一个有序对M=(S,I),S所有的线性基的集合为I,M满足拟阵的性质的证明如下:
1.有穷性:显然存在有限线性基的情况下S是有限的。
2.遗传性:显然一个线性基的任意独立子集都关于异或空间线性无关。(根据线性基定义可得)
3.交换性:
求证:∀A,B∈I不妨设|A|<|B|,必定存在某一元素x∈B-A,使A∪{x}∈I。
证明:
我们假设∀A,B∈I不妨设|A|<|B|,∀x∈B-A,都有A∪{x}∉I。
于是有B与A的差集包含于A的异或空间,又显然B与A的交集包含于A的异或空间,则B包含于A的异或空间。
所以整个B的异或空间包含于A的异或空间,又A,B均线性无关,则有|A|>|B|,与前提|A|<|B|矛盾。命题得证。
引理:带权拟阵的贪心算法正确性证明(懒得写了QAQ,戳这里)
这样我们证明了权值和最大的线性基可以用贪心算法构造,所以按权值排序后贪心加入即可。复杂度O(n(log2n+log2∑numberi))。
【参考代码】
1 #include <algorithm> 2 #include <cstdio> 3 #include <functional> 4 #include <utility> 5 #define REP(i,low,high) for(register int i=(low);i<=(high);++i) 6 #define PER(i,high,low) for(register int i=(high);i>=(low);--i) 7 #define __function__(type) __attribute__((optimize("-O2"))) inline type 8 #define __procedure__ __attribute__((optimize("-O2"))) inline void 9 using namespace std; 10 typedef pair<int,long long> PIL; 11 12 static int n; long long base[64]; PIL a[1010]; 13 14 __function__(bool) push(const long long&n) 15 { 16 long long x=n; PER(i,63,0) if((x>>i)&1) 17 { 18 if(!base[i]) return base[i]=x,1; x^=base[i]; 19 } 20 return 0; 21 } 22 23 int main() 24 { 25 scanf("%d",&n); 26 REP(i,1,n) scanf("%lld%d",&a[i].second,&a[i].first); 27 int ans=0; sort(a+1,a+n+1,greater<PIL>()); 28 REP(i,1,n) ans+=push(a[i].second)*a[i].first; 29 return printf("%d\n",ans),0; 30 }