题意

给你\(n\)个点的树,边有边权

问使得所有的点度数都小于等于\(x\)的最小删边的代价 \([x \in 0...n-1]\)


题解

首先对于每个\(x\)

可以有一个\(O(nlogn)\)的做法

就是设\(f[u][0/1]\)表示不选择/选择点\(u\)的最小代价

那么就是把所有儿子按照\(f[v][1]+w-f[v][0]\)排序

然后\(f[u][0]\)就是选择至少前\(d[u]-x\)小的\(f[v][1]+w\),其他选择\(min(f[v][1]+w,f[v][0])\)

同理,\(f[u][1]\)就是选择至少前\(d[u]-x-1\)小的

其实这个可以通过对于所有儿子都是加上\(f[v][0]\)

然后把\(f[v][1]+w-f[v][0]\)加入小根堆并选择至少\(d[u]-x\)个来实现

然后我们可以发现随着\(x\)的增加,合法的点数越来越少

那么我们可以每次只考虑合法的点,把不合法的点的边权加入父节点中然后删除

然后遍历合法儿子的时候也是把\(f[v][1]+w-f[v][0]\)加入父节点中

那么问题就是找到至少前\(d[u]-x\)小的最小值的和

平衡树即可解决

每次遍历合法点的复杂度是

\(\sum_{i=0}^{n-1}\sum_{j=1}^{n}[i\le d_j]\)

然而一棵树的\(d_i\)和为\(n\times 2-2\)

所以遍历的总复杂度是\(O(n)\)

代码

#include<queue>
#include<vector>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
# define LL long long
const int M = 250005 ;
const LL INF = 1e16 ;
using namespace std ; inline int read() {
char c = getchar() ; int x = 0 , w = 1 ;
while(c>'9'||c<'0') { if(c=='-') w = -1 ; c = getchar() ; }
while(c>='0'&&c<='9') { x = x*10+c-'0' ; c = getchar() ; }
return x*w ;
} int vis[M] ;
int n , num , hea[M] ;
int d[M] , pi[M] , fdis[M] , fa[M] ;
int Tag , dmx , rt[M] ; LL ans , f[M][2] ;
struct Node { int v , w ; } ;
inline bool operator < (Node A , Node B) {
return d[A.v] > d[B.v] ;
}
vector < Node > vec[M] ;
inline bool cmp(int a , int b) {
return d[a] < d[b] ;
}
inline void add_edge(int u , int v , int w) {
vec[u].push_back((Node) { v , w }) ;
} void fdfs(int u , int father) {
fa[u] = father ;
for(int i = 0 , v , w , sz = vec[u].size() ; i < sz ; i ++) {
v = vec[u][i].v , w = vec[u][i].w ; if(v == father) continue ;
fdis[v] = w ; fdfs(v , u) ;
}
}
namespace fhq {
# define ls (son[now][0])
# define rs (son[now][1])
int tot ;
LL sum[M * 8] , val[M * 8] ;
int size[M * 8] , pos[M * 8] , son[M * 8][2] ;
inline int New(LL w) {
size[++tot] = 1 ; pos[tot] = rand() ;
sum[tot] = w ; val[tot] = w ; return tot ;
}
inline void pushup(int now) {
size[now] = size[ls] + size[rs] + 1 ;
sum[now] = sum[ls] + sum[rs] + val[now] ;
}
int Merge(int x , int y) {
if(!x || !y) return x + y ;
if(pos[x] < pos[y]) {
son[x][1] = Merge(son[x][1] , y) ;
pushup(x) ; return x ;
}
else {
son[y][0] = Merge(x , son[y][0]) ;
pushup(y) ; return y ;
}
}
void Split(int now , LL k , int &x , int &y) {
if(!now) return (void)(x = y = 0) ;
if(val[now] <= k) {
x = now ;
Split(rs , k , rs , y) ;
}
else {
y = now ;
Split(ls , k , x , ls) ;
}
pushup(now) ;
}
inline void Insert(int &root , LL w) {
int x , y ;
Split(root , w , x , y) ;
root = Merge(Merge(x , New(w)) , y) ;
}
inline void Del(int &root , LL w) {
int x , y , z ;
Split(root , w , x , z) ;
Split(x , w - 1 , x , y) ;
y = Merge(son[y][0] , son[y][1]) ;
root = Merge(Merge(x , y) , z) ;
}
inline LL Rnk_val(int now , int k) {
while(1) {
if(k <= size[ls]) now = ls ;
else if(k == size[ls] + 1) return val[now] ;
else k -= size[ls] + 1 , now = rs ;
}
}
inline LL Kth_Sum(int now , int k) { // 找前k大元素的和
if(!k) return 0 ; LL ret = 0 ;
while(1) {
if(k <= size[ls]) now = ls ;
else if(k == size[ls] + 1) {
ret += sum[ls] + val[now] ;
return ret ;
}
else {
ret += sum[ls] + val[now] ;
k -= size[ls] + 1 ;
now = rs ;
}
}
}
}
void dfs(int u , int father) {
vis[u] = Tag ; f[u][0] = f[u][1] = 0 ;
if(d[u] <= Tag) return ;
for(int i = 0 , v , w , sz = vec[u].size() ; i < sz ; i ++) {
v = vec[u][i].v , w = vec[u][i].w ;
if(v == father) continue ;
dfs(v , u) ;
f[u][0] += f[v][0] ; f[u][1] += f[v][0] ;
fhq::Insert(rt[u] , f[v][1] + w - f[v][0]) ;
}
int cnt = 0 ; LL x , y , v ;
int l = 1 , r = fhq::size[rt[u]] , ret = 0 , mid ;
while(l <= r) { // 找有几个小于0
mid = (l + r) >> 1 ;
if(fhq::Rnk_val(rt[u] , mid) < 0) ret = mid , l = mid + 1 ;
else r = mid - 1 ;
}
if(ret <= d[u] - Tag)
f[u][0] += fhq::Kth_Sum(rt[u] , d[u] - Tag) ;
else f[u][0] += fhq::Kth_Sum(rt[u] , ret) ;
if(ret <= d[u] - Tag - 1)
f[u][1] += fhq::Kth_Sum(rt[u] , d[u] - Tag - 1) ;
else f[u][1] += fhq::Kth_Sum(rt[u] , ret) ;
for(int i = 0 , v , w , sz = vec[u].size() ; i < sz ; i ++) {
v = vec[u][i].v , w = vec[u][i].w ;
if(v == father) continue ;
fhq::Del(rt[u] , f[v][1] + w - f[v][0]) ;
}
}
int main() {
n = read() ;
for(int i = 1 , u , v , w ; i < n ; i ++) {
u = read() ; v = read() ; w = read() ;
add_edge(u , v , w) ; add_edge(v , u , w) ;
++ d[u] ; ++ d[v] ; ans += w ;
}
fdfs(1 , 0) ;
for(int i = 1 ; i <= n ; i ++) {
pi[i] = i ;
dmx = max( dmx , d[i] ) ;
sort(vec[i].begin() , vec[i].end()) ;
}
sort(pi + 1 , pi + n + 1 , cmp) ;
printf("%lld ",ans) ;
for(int x = 1 , Now = 1 ; x < n ; x ++) {
Tag = x ; ans = 0 ;
while(Now < n && d[pi[Now]] <= x) {
f[pi[Now]][0] = 0 ;
f[pi[Now]][1] = 0 ;
++ Now ;
}
for(int j = Now ; j <= n ; j ++) {
int v ;
while(!vec[pi[j]].empty()) {
v = vec[pi[j]][vec[pi[j]].size() - 1].v ;
if(d[v] <= x) {
if(pi[j] == fa[v])
fhq::Insert( rt[pi[j]] , fdis[v] ) ;
vec[pi[j]].pop_back() ;
}
else break ;
}
}
for(int j = Now , u ; j <= n ; j ++)
if(vis[pi[j]] != x) {
u = pi[j] ;
while(fa[u] && d[fa[u]] > x)
u = fa[u] ;
dfs(u , 0) ;
ans += min(fdis[u] > 0 ? f[u][1] + fdis[u] : INF , f[u][0]) ;
}
printf("%lld ",ans) ;
}
return 0 ;
}

CF1119F Niyaz and Small Degrees的更多相关文章

  1. CF1119F Niyaz and Small Degrees【treedp+堆】

    如果枚举d来dp,那么就是设f[u][0/1]为u点不断/断掉和父亲的边,然后优先选取f[v][1]+w(u,v)<=f[v][0]的,如果断掉这些度数还是多就用一个堆维护剩下的按f[v][1] ...

  2. 树形DP ---- Codeforces Global Round 2 F. Niyaz and Small Degrees引发的一场血案

    Aspirations:没有结果,没有成绩,acm是否有意义?它最大的意义就是让我培养快速理解和应用一个个未知知识点的能力. ————————————————————————————————————— ...

  3. 【codeforces contest 1119 F】Niyaz and Small Degrees

    题目 描述 \(n\) 个点的树,每条边有一个边权: 对于一个 \(X\) ,求删去一些边后使得每个点的度数 \(d_i\) 均不超过 \(X\) 的最小代价: 你需要依次输出 \(X=0 \to n ...

  4. CF 1119F Niyaz and Small Degrees

    打VP的时候由于CXR和XRY切题太快了导致我只能去写后面的题了 然而VP的时候大概还有一小时时想出了\(O(n^2\log n)\)的暴力,然后过了二十分钟才想到删点的优化 结果细节很多当然是写不出 ...

  5. [codeforces contest 1119 F] Niyaz and Small Degrees 解题报告 (树形DP+堆)

    interlinkage: http://codeforces.com/contest/1119/problem/F description: 有一颗$n$个节点的树,每条边有一个边权 对于一个$x$ ...

  6. Solution -「CF 1119F」Niyaz and Small Degrees

    \(\mathcal{Description}\)   Link.   给定一棵 \(n\) 个结点的树,边有边权,对于每个整数 \(x\in[0,n)\),求出最少的删边代价使得任意结点度数不超过 ...

  7. Codeforces Global Round 2 Solution

    这场题目设置有点问题啊,难度:Div.2 A->Div.2 B->Div.2 D->Div.2 C->Div.2 D->Div.1 D-> Div.1 E-> ...

  8. Codeforces Global Round 2 部分题解

    F.Niyaz and Small Degrees 挺sb的一题,为什么比赛时只过了4个呢 考虑当\(x\)固定的时候怎么做.显然可以树形DP:设\(f_{u,i=0/1}\)表示只考虑\(u\)子树 ...

  9. [TimusOJ1057]Amount of Degrees

    [TimusOJ1057]Amount of Degrees 试题描述 Create a code to determine the amount of integers, lying in the ...

随机推荐

  1. cds.data:=dsp.data赋值有时会出现AV错误剖析

    cds.data:=dsp.data赋值有时会出现AV错误剖析 如果QUERY没有查询到任何数据,cds.data:=dsp.data赋值会触发AV错误. 大家知道,DATASNAP有许多远程方法就是 ...

  2. 使用图像扫描控件ScanOnWeb实现在线图像扫描

    今天上网查资料,看到一篇文章,描述的是一个开发OA软件的公司解决浏览器嵌入式扫描仪编程的文章,文章描述了改OA厂商的工程师如何辛苦的克服了各种技术难题,最终实现了在线图像扫描处理,然后又在无数个不眠的 ...

  3. Go --- 设计模式(模板模式)

    模版模式真的是一个好东西.所谓模版模式,就是说,某几个类中相同的操作和代码提取到父类的一个函数中,并定义相同的操作为抽象函数.由子类来实现.估计我也没表达清楚,下面还是看代码来讲解吧. 例:我们有两个 ...

  4. 关于对FLASH开发,starling、starling feathers、starling MVC框架的理解

    说在前头:楼主之前没有不论什么flash开发经验,仅仅是从一次尝试中总结自己的理解和经验而已.假设有写的不正确的地方,欢迎大家指正. 前一段时间尝试想用flash(as3)又一次制作一下之前做的一个游 ...

  5. 以Java属性文件的格式创建Hibernate的配置文件和DTD特殊符号作用

    演示样例代码 hibernate.dialect=org.hibernate.dialect.MySQLDialect hibernate.connection.driver_class=com.my ...

  6. 轻量级批量Omnitty工具安装和简单使用

    一.Omnitty简单介绍 在实际工作需要同时对多台docker进行批量处理,为了节省时间,这个运维轻量级工具解决此问题 二.Omnitty简单按照 下载需要安装包和依赖包: omnitty-0.3. ...

  7. 嵌入式开发之davinci---DM8168 8127 8148 HDVPSS中的一些英文缩写解释

    BLEND:Alpha blends input with the graphics.将输入的视频与图形做Alpha融合. CPROC:Color Processing.颜色处理.如动态对比度增强.饱 ...

  8. 【leetcode】Word Break(python)

    思路是这种.我们从第一个字符開始向后依次找,直到找到一个断句的地方,使得当前获得的子串在dict中,若找到最后都没找到.那么就是False了. 在找到第一个后,接下来找下一个断句处,当然是从第一个断句 ...

  9. The server committed a protocol violation. Section=ResponseHeader Detail=CR must be followed by LF

    打开表单偶尔会出现这个提示,解决方法: web.config增加配置: <configuration> <system.net> <settings> <ht ...

  10. socket.io中文文档

    socket.io 中文文档转载于:http://www.cnblogs.com/xiezhengcai/p/3956401.html 服务端 io.on(‘connection’,function( ...