Description

给定一颗 \(n\) 个结点的树,每个点有一个点权 \(v\)。点权只可能为 \(0\) 或 \(1\)。

现有一个空数列,每次可以向数列尾部添加一个点 \(i\) 的点权 \(v_i\),但必须保证此时 \(i\) 没有父结点。添加后将 \(i\) 删除。

这样可以一个长为 \(n\) 的数列 \(x\)。求 \(x\) 中逆序对数的最小值。

Hint

  • \(1\le n\le 2\times 10^5\)
  • \(v_i \in \{0, 1\}\)

Solution

由于一个结点的父结点尚未被删除,那么现在该结点则无法被加入数列。可见题目要求我们 从树根自顶向下 删除。

但显然我们不会这样做——我们 将所有结点视作独立,向父亲方向合并


我们不妨先考虑这样一个问题:对于一个根结点为 \(x\) 的树,其子结点为 \(y_1, y_2, \cdots y_k\)。假设子树 \(y_1, y_2, \cdots y_k\) 都已经合并好了,那么我们只要将这些子树合并答案,向上传答案即可。

首先,由题意得,结点 \(x\) 的点权必须排在最前面。接下来就需要合理安排顺序,使得 跨越子树的逆序对 数量最小。由于子树内在前期早已统计完毕,此处无需再做讨论。

为方便讨论,在这里我们还需维护子树中 \(0, 1\) 的个数,分别记为 \(\text{cnt}(\cdots, 0), \text{cnt}(\cdots, 1)\)。

若要使逆序对尽可能小,而权值就只有 \(0, 1\),第一直觉就是 贪心地把 \(0\) 尽量排前面

但直觉是很模糊的,我们需要一个明确的标准。

对于两个子树 \(y_i, y_j\),如果 \(y_i\) 排在前面,那么会产生 \(\text{cnt}(y_i, 1)\times \text{cnt}(y_j, 0)\) 个逆序对,反正则会产生 \(\text{cnt}(y_j, 1)\times \text{cnt}(y_i, 0)\) 个。

显然我们应选择结果较少的策略——优先选取 \(\dfrac{\text{cnt}(y, 0)}{\text{cnt}(y, 1)}\) 较小的。为避免除以零造成 RE,需要化除为乘。


但此题不能直接递归处理,需要全局一起算,即上文中“将所有结点视作独立,向父亲方向合并”的思路。

那么子树的 \(\text{cnt}\) 值就变成了 连通块 的 \(\text{cnt}\) 值,容易发现上面的贪心思路于此仍然有效。

此处涉及连通块整块信息的维护,不难想到 并查集。连通块的有序维护,可以使用

在每个点向上合并后,父亲方向结点需要删去,这对于堆来说就不太方便(当然可以考虑 multiset 或 可删堆

但其实不用这么麻烦:直接根据 \(\text{cnt}\) 值判断是否已经被合并然后选择性跳过即可。

最后做到 1 号点就不用重新插入堆中了。

Code

/*
* Author : _Wallace_
* Source : https://www.cnblogs.com/-Wallace-/
* Problem : AtCoder AGC023F 01 on Tree
*/
#include <algorithm>
#include <iostream>
#include <queue> using namespace std;
const int N = 2e5 + 5; int n, fa[N], dsu[N];
int cnt[N][2]; struct item {
int c0, c1, idx;
bool operator < (const item& t) const {
return c0 * 1ll * t.c1 < c1 * 1ll * t.c0;
}
};
priority_queue<item> pq; int find(int x) {
return x == dsu[x] ? x : dsu[x] = find(dsu[x]);
} signed main() {
ios::sync_with_stdio(false); cin >> n;
for (register int i = 2; i <= n; i++)
cin >> fa[i];
for (register int i = 1, val; i <= n; i++)
cin >> val, cnt[i][val]++;
for (register int i = 1; i <= n; i++)
dsu[i] = i; long long ans = 0;
for (register int i = 2; i <= n; i++)
pq.push({cnt[i][0], cnt[i][1], i}); while (!pq.empty()) {
item cur = pq.top(); pq.pop();
int x = find(cur.idx), c0 = cur.c0, c1 = cur.c1; if (cnt[x][0] != c0 || cnt[x][1] != c1)
continue; int y = find(fa[x]);
ans += cnt[y][1] * 1ll * cnt[x][0];
cnt[y][0] += cnt[x][0];
cnt[y][1] += cnt[x][1]; dsu[x] = y;
if (y > 1) pq.push({cnt[y][0], cnt[y][1], y});
} cout << ans << endl;
return 0;
}

【AtCoder AGC023F】01 on Tree(贪心)的更多相关文章

  1. 【luogu AT3957】[AGC023F] 01 on Tree

    01 on Tree 题目链接:luogu AT3957 题目大意 有一棵根为 \(1\) 的树,每个节点有个值 \(0\) 或 \(1\). 然后每次你可以把一个没有父亲的点删除,然后把值放进一个数 ...

  2. [atcoder contest 010] F - Tree Game

    [atcoder contest 010] F - Tree Game Time limit : 2sec / Memory limit : 256MB Score : 1600 points Pro ...

  3. AtCoder Grand Contest 023 F - 01 on Tree

    Description 题面 Solution HNOI-day2-t2 复制上去,删点东西,即可 \(AC\) #include<bits/stdc++.h> using namespa ...

  4. D - Digging(01背包,贪心)

    D - Digging Time Limit:2000MS     Memory Limit:65536KB     64bit IO Format:%lld & %llu Submit St ...

  5. AtCoder ABC 070D - Transit Tree Path

    传送门:http://abc070.contest.atcoder.jp/tasks/abc070_d 本题是一个图论问题——树(Tree). 有一棵结点数目为n的无向树.第i条边连接结点ai与bi, ...

  6. $POJ$2976 $Dropping\ tests$ 01分数规划+贪心

    正解:01分数规划 解题报告: 传送门! 板子题鸭,,, 显然考虑变成$a[i]-mid\cdot b[i]$,显然无脑贪心下得选出最大的$k$个然后判断是否大于0就好(,,,这么弱智真的算贪心嘛$T ...

  7. HDU 4825 Xor Sum(经典01字典树+贪心)

    Xor Sum Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 132768/132768 K (Java/Others) Total ...

  8. Poj 2499 Binary Tree(贪心)

    题目链接:http://poj.org/problem?id=2499 思路分析:结点向左边移动时结点(a, b)变为( a+b, b),向右边移动时( a, b )变为( a, a + b); 为求 ...

  9. POJ 2054 Color a Tree#贪心(难,好题)

    题目链接 代码借鉴此博:http://www.cnblogs.com/vongang/archive/2011/08/19/2146070.html 其中关于max{c[fa]/t[fa]}贪心原则, ...

随机推荐

  1. WIN10下安装python3.7.2出现“尝试创建C:\Users\XX\AppData\Roaming\Microsoft\Installer时出错”

    WIN10下安装python3.7.2出现"尝试创建C:\Users\XX\AppData\Roaming\Microsoft\Installer时出错" 1.右键点击安装包以管理 ...

  2. nginx配置代理缓存

    nginx可以实现反向代理的配置,并且可以使用缓存来加速,本文是简单的实现功能的配置,暂时没有做其他的优化的部分的配置,从网上的资料来看,很多配置都是没有讲哪些是必须配置的,我自己在配置过程中就发现没 ...

  3. Linux下如何创建loop device

    在Linux中,有一种特殊的块设备叫loop device,这种loop device设备是通过映射操作系统上的正常的文件而形成的虚拟块设备 因为这种设备的存在,就为我们提供了一种创建一个存在于其他文 ...

  4. Tomcat口令暴力猜解&&后台getshell

    Tomcat环境搭建 windows系统xampp搭建tomcat linux yum搭建tomcat 修改tomcat目录下的conf/tomcat-users.xml文件开启管理后台口令认证 &l ...

  5. DWVA-XSS部分练手闯关

    前言 关于XSS基础内容请查看:https://www.cnblogs.com/xhds/p/12239527.html 实验平台采用DWVA  v1.10 XSS(Reflected)反射性XSS漏 ...

  6. 推荐系统实践 0x05 推荐数据集MovieLens及评测

    推荐数据集MovieLens及评测 数据集简介 MoiveLens是GroupLens Research收集并发布的关于电影评分的数据集,规模也比较大,为了让我们的实验快速有效的进行,我们选取了发布于 ...

  7. 教你在CorelDRAW中制作水印

    水印是一种数字保护的手段,在图像上添加水印即能证明本人的版权,还能对版权的保护做出贡献.也就是在图片上打上半透明的标记,因其具有透明和阴影的特性,使之不管在较为阴暗或明亮的图片上都能完美使用,嵌入的水 ...

  8. 初学者也能轻松做出好Beat:FPC鼓机使用教程

    如果我们想用FL Studio制作一个鼓的声部,这时水果自带的鼓机FPC简直就是我们初学者的福音.因为它的操作比较简单,自带的鼓谱也很丰富,而且我们还可以对鼓的音色做细致的调整,或者是使用自己的采样替 ...

  9. 基于Python+requests搭建的自动化框架-实现流程化的接口串联

    框架产生目的:公司走的是敏捷开发模式,编写这种框架是为了能够满足当前这种发展模式,用于前后端联调之前(后端开发完接口,前端还没有将业务处理完毕的时候)以及日后回归阶段,方便为自己腾出学(mo)习(yu ...

  10. 用大白话讲大数据HBase,老刘真的很用心(1)

    老刘今天复习HBase知识发现很多资料都没有把概念说清楚,有很多专业名词一笔带过没有解释.比如这个框架高性能.高可用,那什么是高性能高可用?怎么实现的高性能高可用?没说! 如果面试官听了你说的,会有什 ...