根据题目可知,答案是若干个区间异或和的和。若进行异或前缀和,记
首先,对于每个右端点
然后就做完了。
AC Code:
#include <bits/stdc++.h>
using namespace std;
namespace Slongod{
using uint = unsigned int;
using ll = long long int;
const int N = 5e5+7;
//可持久化tri树
#define ls tree[ro].son[1]
#define rs tree[ro].son[0]
struct TREE{int size , id , son[2];}tree[N*33];
int cnt , head[N];
int copyfrom(int ro){cnt++; tree[cnt] = tree[ro]; return cnt;}
void check(int &ro){ro = (!ro ? ++cnt : copyfrom(ro));}
void update(int ro , uint x , int id , int d=31)
{
if (d < 0){tree[ro].size++; tree[ro].id = id; return;}
int c = ((x>>d)&1);
check(tree[ro].son[c]);
update(tree[ro].son[c] , x , id , d-1);
tree[ro].size = tree[ls].size + tree[rs].size;
}
pair<uint,int> query(int lo , int ro , uint x , int d=31)
{
if (d < 0){return {0,tree[ro].id};}
int c = ((x>>d)&1);
if (tree[ro].son[!c] and tree[tree[ro].son[!c]].size - tree[tree[lo].son[!c]].size) {
auto t = query(tree[lo].son[!c] , tree[ro].son[!c] , x , d-1);
return {(uint(1)<<d)|(t.first) , t.second};
} else {
return query(tree[lo].son[c] , tree[ro].son[c] , x , d-1);
}
}
#undef ls
#undef rs
struct node
{
uint val , op; int l , r , c;
friend bool operator < (node a , node b){return a.val < b.val;}
};
int n , k; uint a[N]; ll ans;
int main()
{
cin >> n >> k; n++; for (int i = 2; i <= n; i++){cin >> a[i]; a[i] ^= a[i-1];}
priority_queue <node> q;
for (int i = 1; i <= n; i++) {
head[i] = copyfrom(head[i-1]);
update(head[i] , a[i] , i);
if (i > 1) {
auto t = query(head[0] , head[i-1] , a[i]);
q.push({.val=t.first , .op=a[i] , .l=1 , .r=i-1 , .c=t.second});
}
}
for (int i = 1; i <= k; i++) {
auto t = q.top(); q.pop();
ans += t.val;
if (t.c != t.l) {
auto tmp = query(head[t.l-1] , head[t.c-1] , t.op);
q.push({.val=tmp.first , .op=t.op , .l=t.l , .r=t.c-1 , .c=tmp.second});
}
if (t.c != t.r) {
auto tmp = query(head[t.c] , head[t.r] , t.op);
q.push({.val=tmp.first , .op=t.op , .l=t.c+1 , .r=t.r , .c=tmp.second});
}
}
cout << ans << '\n';
return 0;
}
}int main()
{
#ifndef ONLINE_JUDGE
freopen("in.in" , "r" , stdin);
#endif
ios :: sync_with_stdio(0);
cin.tie(0) , cout.tie(0);
return Slongod :: main();
}