题目描述

给定一棵\(n\)个点的有根树,根节点编号为\(1\),点有点权。

定义\(d(v)\)表示\(v\)到\(1\)的路径上的边数。

定义\(f(v,u)\)在\(v<u\)且\(v\)和\(u\)任意一个都不是另一个的祖先时为\(1\),否则为\(0\)。

定义\(g(v,u)\)在\(v\)是\(u\)的祖先且\(v\)的权值大于\(u\)的权值时为\(1\),否则为\(0\)。

定义\(h(v,u)\)在\(v\)是\(u\)的祖先且\(v\)的权值小于\(u\)的权值时为\(1\),否则为\(0\)。

你需要将点集分成两个集合\(A\)和\(B\),有\(m\)组询问,每组询问给定了集合\(A\)的大小,求下列表达式的最小值:

\[F(A,B)=\sum_{v\in A,u\in A,v\not=u}f(v,u)+g(v,u)+\sum_{v\in B,u\in B,v\not = u} h(v,u)+\sum_{v\in A}d(v)
\]

输入

第一行两个整数\(n,m\)。

接下来一行\(n\)个整数\(a_i\)表示第\(i\)个点的权值。

接下来\(n−1\)行,第\(i\)行两个整数\(v,u\)表示一条连接\((v,u)\)的边。

接下来\(m\)行,每行一个整数表示\(|A|\)。

输出

共\(m\)行,每行一个整数表示表达式的最小值。

样例

样例输入

4 3
4 1 2 3
1 2
2 3
2 4
0
2
4

样例输出

2
2
9

数据范围

对于\(100\%\)的数据,\(1\leq n,m\leq 500000,0\leq |A|\leq n,1\leq a_i\leq 500000\)。

比第二题又难了不少……(差评,题目难度指数式上升)

这一坨定义看起来好难受啊,考虑转化一下。

首先看\(f\)的意义,发现其实就是\(|A|\)中互不为祖先的点对数,等价于\(C(|A|,2)-\)每个点在\(A\)中的祖先个数的和;\(d\)的意义是深度,也就是\(A\)中所有点在树上的祖先个数的和。

那这两个加起来是什么呢?就是\(C(|A|,2)+\)满足\(u\in A,v\in B\)且\(u\)是\(v\)的祖先的点对数量。

另外,我们定义一个新函数\(e(u,v)\)当\(u\)是\(v\)的祖先,且\(a_u=a_v\)时为\(1\),否则为\(0\)。显然\([u\)是\(v\)的祖先\(]\)等价于\(g(u,v)+h(u,v)+e(u,v)\)。那么化一下原式:

\[F(A,B)=C(|A|,2)+\sum_{v\in A,u\in A,v\not=u}g(v,u)+\sum_{v\in B,u\in B,v\not = u} h(v,u)+\sum_{v\in A,u\in B}g(v,u)+h(v,u)+e(v,u)
\]

把\(g\)和\(h\)合并到前面去,于是\(g\)和\(h\)的一半部分变成全集了:

\[F(A,B)=C(|A|,2)+\sum_{v\in A}g(v,u)+\sum_{u\in B} h(v,u)+\sum_{v\in A,u\in B}e(v,u)
\]

然后还有一个\(e\)比较烦人,注意到若\(v\)是\(u\)的祖先,且\(v\in B,u\in A\),如果交换\(v\)和\(u\),\(g\)部分会变更优,\(h\)部分会变更优,\(e\)部分也会变更优,当然会优先选择祖先……所以当一个点被加入\(A\)中时,他的祖先里和它权值一样的一定早就全都被加入到里面去了。所以\(e\)可以很轻松地计算,此时每个点加入A中带来的权值已经和其他元素毫不相干了,只需要用树状数值帮助预处理一下即可。

\(Code:\)

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
#define N 500005
#define M 1000005
#define ll long long
ll ans[N];
int n, m, A[N], tre[N], sum[N];
int tar[M], nex[M], fir[N], cnt, tim;
int fat[N], dep[N], dfn[N], out[N], val[N], idx[N];
void Read(int &p)
{
p = 0;
char c = getchar();
for (; c < '0' || c > '9'; c = getchar());
for (; c >= '0' && c <= '9'; c = getchar())p = p * 10 + c - '0';
}
void Update(int x, int v)
{
for (int i = x; i <= N - 5; i += (i & -i))
tre[i] += v;
}
int Getsum(int x)
{
int ans = 0;
for (int i = x; i; i -= (i & -i))
ans += tre[i];
return ans;
}
void Add(int u, int v)
{
++cnt;
tar[cnt] = v;
nex[cnt] = fir[u];
fir[u] = cnt;
}
void Dfs(int r)
{
dfn[r] = ++tim;
val[r] = dep[r] - Getsum(A[r]);
Update(A[r], 1);
for (int i = fir[r]; i; i = nex[i])
{
int v = tar[i];
if (v != fat[r])
{
fat[v] = r;
dep[v] = dep[r] + 1;
Dfs(v);
}
}
Update(A[r], -1);
out[r] = tim;
};
bool cmp(int a, int b){return A[a] > A[b];}
int main()
{
scanf("%d%d", &n, &m);
for (int i = 1; i <= n; i++)
Read(A[i]), idx[i] = i;
for (int i = 1; i < n; i++)
{
int u, v;
Read(u), Read(v);
Add(u, v), Add(v, u);
}
Dfs(1);
sort(idx + 1, idx + n + 1, cmp);
for (int i = 1; i <= n; )
{
int x = i;
while (x <= n && A[idx[i]] == A[idx[x]])
x++;
for (int k = i; k < x; k++)
sum[idx[k]] = Getsum(out[idx[k]]) - Getsum(dfn[idx[k]] - 1);
for (int k = i; k < x; k++)
Update(dfn[idx[k]], 1);
i = x;
}
for (int i = 1; i <= n; i++)
val[i] -= sum[i], ans[0] += sum[i];
sort(val + 1, val + n + 1);
for (int i = 1; i <= n; i++)
ans[i] = ans[i - 1] + val[i] + i - 1;
for (; m--; )
{
int x;
Read(x);
printf("%lld\n", ans[x]);
}
}

「模拟赛20191019」C 推式子+贪心+树状数组的更多相关文章

  1. 「模拟赛20191019」B 容斥原理+DP计数

    题目描述 将\(n\times n\)的网格黑白染色,使得不存在任意一行.任意一列.任意一条大对角线的所有格子同色,求方案数对\(998244353\)取模的结果. 输入 一行一个整数\(n\). 输 ...

  2. 「模拟赛20191019」A 简单DP

    题目描述 给一个\(n\times m\)的网格,每个格子上有一个小写字母. 对于所有从左上角\((1,1)\)到右下角\((n,m)\)只向下或向右走的路径构成的集合,判断是否存在两条走法不同的路径 ...

  3. 「模拟赛20180306」回忆树 memory LCA+KMP+AC自动机+树状数组

    题目描述 回忆树是一棵树,树边上有小写字母. 一次回忆是这样的:你想起过往,触及心底--唔,不对,我们要说题目. 这题中我们认为回忆是这样的:给定 \(2\) 个点 \(u,v\) (\(u\) 可能 ...

  4. 「模拟赛20180307」三元组 exclaim 枚举+树状数组

    题目描述 给定 \(n,k\) ,求有多少个三元组 \((a,b,c)\) 满足 \(1≤a≤b≤c≤n\)且\(a + b^2 ≡ c^3\ (mod\ k)\). 输入 多组数据,第一行数据组数\ ...

  5. [CSP-S模拟测试]:斯诺(snow)(数学+前缀和+树状数组)

    题目传送门(内部题37) 输入格式 第一行一个整数$n$,表示区间的长度. 第二行一个长度为$n$的只包含$0,1,2$的字符串,表示给出的序列. 输出格式 一行一个整数,表示革命的区间的数量. 样例 ...

  6. 2019ICPC 上海网络赛 L. Digit sum(二维树状数组+区间求和)

    https://nanti.jisuanke.com/t/41422 题目大意: 给出n和b,求1到n,各数在b进制下各位数之和的总和. 直接暴力模拟,TLE.. 没想到是要打表...还是太菜了. # ...

  7. 「模拟赛20190327」 第二题 DP+决策单调性优化

    题目描述 小火车虽然很穷,但是他还是得送礼物给妹子,所以他前往了二次元寻找不需要钱的礼物. 小火车准备玩玩二次元的游戏,游戏当然是在一个二维网格中展开的,网格大小是\(n\times m\)的,某些格 ...

  8. 「模拟赛20181025」御风剑术 博弈论+DP简单优化

    题目描述 Yasuo 和Riven对一排\(n\)个假人开始练习.斩杀第\(i\)个假人会得到\(c_i\)个精粹.双方轮流出招,他们在练习中互相学习,所以他们的剑术越来越强.基于对方上一次斩杀的假人 ...

  9. 「模拟赛20180406」膜树 prufer编码+概率

    题目描述 给定一个完全图,保证\(w_{u,v}=w_{v,u}\)且\(w_{u,u}=0\),等概率选取一个随机生成树,对于每一对\((u,v)\),求\(dis(u,v)\)的期望值对\(998 ...

随机推荐

  1. 一行代码搞定WordPress面包屑导航breadcrumb

    有好几位网友在问WordPress面包屑导航breadcrumb怎么操作,网上有些教程是去function文件中定义,其实不用那么复杂,很简单一行代码就能搞定.下面随ytkah一起来看看.如果是单页, ...

  2. 学习Spring-Data-Jpa(十)---注解式方法查询之@Query、@Modifying与派生delete

    1.@Query 对于少量的查询,使用@NamedQuery在实体上声明查询是一种有效的办法,并且可以很好的工作.由于查询本身绑定到执行它们的java方法,实际上可以通过Spring-Data-Jpa ...

  3. siameseNet网络以及信号分类识别应用

    初学siameseNet网络,希望可以用于信号的识别分类应用.此文为不间断更新的笔记. siameseNet简介 全连接孪生网络(siamese network)是一种相似性度量方法,适用于类别数目多 ...

  4. WinDbg常用命令系列---显示引用的内存(dda、ddp、ddu、dpa、dpp、dpu、dqa、dqp、dqu)

    命令dda, ddp, ddu, dpa, dpp, dpu, dqa, dqp, 和 dqu在指定位置显示指针,取消对该指针的引用,然后以各种格式显示结果位置的内存. ddp [Options] [ ...

  5. Log4net 单独创建配置文件(三)

    1.建立ASP.Net空的Web程序,添加Default.aspx窗体 2.添加web配置文件命名为:log4net.config,添加配置 <?xml version="1.0&qu ...

  6. web前端开发面试被虐篇(一)

    地点:北京 职位:前端开发工程师 要求:达到中级开发,JS基础足够扎实,css基础扎实,要求纯手写代码 面试过程: 进门一个面相老成的大叔递给我一份题,说别的都不好使先做题看看水平,说话语气很温和明显 ...

  7. 【loj2983】【WC2019】数树

    题目 两颗\(n\)个点的树T1和T2,有\(y\)种颜色; 现在给每个点染色,要求公共边端点的颜色相同,求: ​ 1.op=0 , T1和T2都确定,求合法染色方案数: ​ 2.op=1 , T1确 ...

  8. D3.js的v5版本入门教程(第十一章)——交互式操作

    D3.js的v5版本入门教程(第十一章) 与图形进行交互操作是很重要的!所谓的交互操作也就是为图形元素添加监听事件,比如说当你鼠标放在某个图形元素上面的时候,就会显示相应的文字,而当鼠标移开后,文字就 ...

  9. ICEM-哑铃(无厚度)

    原视频下载地址:https://pan.baidu.com/s/1i44hdkh 密码: 96dh

  10. vue-route开发注意事项

    1.router-link to 动态赋值 router-link组件的to属性值有两种写法,字符串类型和对象类型 如果要动态传值,比如放到for循环中,传入for的index值,这时就必须使用对象形 ...