• P4570 [BJWC2011]元素 (线性基)


    题意:n个石头 每个石头有a,b两个属性

       要求选出一些石头使得没有一个子集的a属性xor和为0 且b属性和最大

    题解:线性基例题了.. 好像需要理解一些性质

       1.原序列里任一数都可有由线性基xor得到

       2.线性基里的数是线性无关的 及没有一个子集xor和为0 (就刚好满足题意了

       3.线性基在保证性质1的前提下 数的大小是最少的

       

       于是这个题就把b属性从大到小排序 贪心的如果这个数能插进去就算上贡献

       如果某个数插入不进来 那么说明他和之前的插入进来的数 线性相关了 那么之前的数都比他大 那么肯定要前面的数

       然后也不会有先插入了一个大的数A 导致较小的两个B,C插入不进来 但B+C的b属性大于A和A后面选的 这种情况

       假如这样的话 说明A可以由 B,C这种情况给线性表示 那么你显然可以去掉B,C里面一个与A线性相关的点 然后把A放进来

       这样构成一个线性基肯定更优 所以大的数能插就插进来

    #include <bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    
    struct node{
        ll a;
        int b;
    }e[1005];
    bool cmp(node A, node B) {
        return A.b > B.b;
    }
    ll val[70];
    
    int main() {
        int n;
        scanf("%d", &n);
        for(int i = 1; i <= n; i++) scanf("%lld%d", &e[i].a, &e[i].b);
        sort(e + 1, e + 1 + n, cmp);
    
        int ans = 0;
        for(int i = 1; i <= n; i++) {
            ll tmp = e[i].a;
            for(int j = 60; j >= 0; j--) {
                if(tmp & (1LL << j)) {
                    if(!val[j]) {
                        ans += e[i].b;
                        val[j] = tmp;
                        break;
                    }
                    tmp ^= val[j];
                }
            }
        }
        printf("%d
    ", ans);
        return 0;
    }
    View Code
  • 相关阅读:
    tornado硬件管理系统-数据存储与表格实时监控(8)
    db2 内存研究
    Oracle dml开始到commit期间的流程
    用户界面与业务逻辑的分离
    计算器核心算法——终结版
    计算器核心算法——中缀表达式转为后缀表达式
    计算器核心解析算法(上)
    Qt中的字符串类
    初探Qt中的消息处理
    计算器界面代码重构
  • 原文地址:https://www.cnblogs.com/lwqq3/p/11432837.html
Copyright © 2020-2023  润新知