zoukankan      html  css  js  c++  java
  • luogu5283 异或粽子

    题目链接

    思路

    首先求个前缀异或和,这样就可以(O(1))的得到区间异或和了。

    然后发现问题转化为

    找出不同的(k)个二元组(x,y)。使得(a_x otimes a_y)的和最大。

    有个比较有趣的思路

    (S_i)表示前(i)个元素的异或和。对于每个(S_i),我们找出在(S)数组中与他异或起来最大的数字是多少。假设第(i)个得到的最大异或和为(t_i)

    然后从这些数字中找出最大的那个。假设是(t_x)。然后我们就把答案加上(t_x),并且把(t_x)变为与(S_x)异或起来第(2)大的异或和。一直这样做下去。

    发现可以用堆维护。

    发现对于每个异或和都被算了两次。所以把(K)先乘(2),并且把最终答案除以(2),就可以了。

    代码

    /*
    * @Author: wxyww
    * @Date:   2019-04-11 19:00:05
    * @Last Modified time: 2019-04-11 19:28:46
    */
    #include<cstdio>
    #include<iostream>
    #include<cstdlib>
    #include<cstring>
    #include<algorithm>
    #include<queue>
    #include<vector>
    #include<ctime>
    using namespace std;
    typedef long long ll;
    const int N = 500000 + 100;
    ll read() {
    	ll x=0,f=1;char c=getchar();
    	while(c<'0'||c>'9') {
    		if(c=='-') f=-1;
    		c=getchar();
    	}
    	while(c>='0'&&c<='9') {
    		x=x*10+c-'0';
    		c=getchar();
    	}
    	return x*f;
    }
    struct node {
    	ll id;
    	int rk;
    	ll w;
    	node(int x,int y,ll z) {
    		id = x,rk = y,w = z;
    	}
    };
    int trie[N * 32][2],cnt[N * 32],tot;
    ll a[N];
    bool operator < (const node &A,const node &B) {
    	return A.w < B.w;
    }
    priority_queue<node>q;
    void insert(ll x) {
    	int now = 0;
    	for(int i = 31;i >= 0;--i) {
    		int k = (x >> i) & 1;
    		if(!trie[now][k]) trie[now][k] = ++tot;
    		now = trie[now][k];
    		cnt[now]++;
    	}
    }
    ll query(ll x,int y) {
    	int now = 0;
    	ll ret = 0;
    	for(int i = 31;i >= 0;--i) {
    		int k = (x >> i) & 1;
    		if(!trie[now][k ^ 1]) now = trie[now][k];
    		else {
    			if(y > cnt[trie[now][k ^ 1]]) y -= cnt[trie[now][k ^ 1]],now = trie[now][k];
    			else now = trie[now][k ^ 1],ret |= (1ll << i);
    		}
    	}
    	return ret;
    }
    int main() {
    	ll ans = 0;
    	int n = read(),K = read() << 1;
    	insert(0);
    	for(int i = 1;i <= n;++i) {
    		a[i] = a[i - 1] ^ read();
    		insert(a[i]);
    	}
    	for(int i = 1;i <= n;++i) q.push(node(a[i],1,query(a[i],1)));
    	q.push(node(0,1,query(0,1)));
    	while(K--) {
    		node tmp = q.top();q.pop();
    		ans += tmp.w;
    		tmp.rk++;
    		tmp.w = query(tmp.id,tmp.rk);
    
    		q.push(tmp);
    	}
    	cout<<(ans >> 1);
    	return 0;
    }
    
  • 相关阅读:
    【模式分解】无损连接&保持函数依赖
    【范式与函数依赖】3NF与BCNF的区别
    C#设置按钮三态背景图片
    C#代码设置窗体和Panel的位置大小
    C#窗体嵌套
    C#中弹出新窗口
    C#定义委托函数实现在别的窗体中操作主窗体中的SerialPort控件
    C#界面设计疑问2:panel摆放问题
    C#拖动自己的定义标题栏(panel)以及实现窗体拖动关闭和最小化
    C#界面设计疑问
  • 原文地址:https://www.cnblogs.com/wxyww/p/luogu5283.html
Copyright © 2011-2022 走看看