Loading

2024杭电多校第9场

9

1001 树异或价值 (hdu7529

对价值定义式进行转化,\((a_i\oplus a_j)\times lca(a_i, a_j)\) 可视为 \(a_i,a_j\) 的所有祖先下 \(\sum a_i\oplus a_j\),数组 \(a\) 总价值即各节点子树中任意两个子节点的异或之和。由按位异或的性质,不同二进制位之间答案互不影响,可简化考虑最低位即 \(a_i\in \{0, 1\}\),计算出此时的 \(ans'\),有 \(ans = (ans')^k\).

\(a_i\in \{0, 1\}\) 时,为了使异或之和尽可能大,即各节点子树内 \((\sum [a=0])\times (\sum [a=1])\) 最大,应使 \(a = 0\)\(a = 1\) 子节点数量尽可能平均;实际上通过构造可实现所有节点下 \(\mid\sum[a = 0] - \sum[a = 1])\mid\space \leq 1\). 设 \(dp_x\)\(a_x = 0\) 时子树符合条件的答案数,\(y\in subtree(x)\)\(se_x,so_x\)分别代表大小为偶数/奇数的 \(y\) 数量,则有:

\(dp_x =\prod dp_y \times 2^{se_x}\times (\dbinom{so_x}{\lfloor\frac{so_x}{2}\rfloor} + \dbinom{so_x}{\frac{so_x}{2} - 1}\times [2|so_x])\).

最终答案即 \((2\times dp_1)^k\). 组合数学的逻辑多少有点抽象,建个模理解一下式子就好,至少代码不难写。

void pre(int i) {
    sz[i] = 1;
    s0[i] = s1[i] = 0;
    for(int j = 0; j < v[i].size(); j++) {
        int t = v[i][j];
        pre(t);
        sz[i] += sz[t];
        if(sz[t] & 1) s1[i]++;
        else s0[i]++;
    }
}
void cal(int i) {
    dp[i] = 1;
    for(int j = 0; j < v[i].size(); j++) {
        int t = v[i][j];
        cal(t);
        dp[i] *= dp[t];
        dp[i] %= mo;
    }
    dp[i] *= fp(2, s0[i]);
    dp[i] %= mo;
    dp[i] *= (c(s1[i], s1[i] / 2) + ((s1[i] & 1) == 0) * c(s1[i], s1[i] / 2 - 1)) % mo;
    dp[i] %= mo;
}

1003 黑洞合并 (hdu7531

真抽象啊,这个《规律》是赛时我发现了都不敢提出来的程度。把它写这里纯粹是为了纪念我写过这么一道抽象的题目()

1006 融合矿石 (hdu7534

非常巧妙的dp. 数据保证随着金辉石占比的增加,矿石价值单调不减,且矿石的数量无限,因此用完全背包预处理出所有可能的融合情况下、一定质量矿石的最大金辉石含量,将结果看作 \(m\) 种不同的新矿石,再用一次完全背包计算即可。

    for(int i = 1; i <= n; i++) {
        int a, b;
        scanf("%d%d", &a, &b);
        for(int j = a; j <= m; j++) {
            f[j] = max(f[j], f[j - a] + b);
        }
    }
    for(int i = 1; i <= m; i++) {
        if(!f[i]) continue;
        int x = (f[i] * 10 - 1) / i + 1;
        f[i] = v[x] * i;
    }
    for(int i = 1; i <= m; i++) {
        for(int j = i; j <= m; j++) {
            g[j] = max(g[j], g[j - i] + f[i]);
        }
    }
    printf("%lld\n", g[m]);
posted @ 2024-08-20 13:41  Aderose_yr  阅读(108)  评论(0编辑  收藏  举报