对于确定的$K$,问题也可以看作每一个点最多选$K$条出边,并最大化选择的边权和

关于这个问题,有如下的树形dp——

令$f_{k,0/1}$表示以$k$为根的子树中,根节点选择不超过$K/K-1$个儿子的最大边权和,转移为
$$
\begin{cases}f_{k,0}=\sum_{x\in son_{k}}f_{x,0}+\max_{S\subseteq son_{k},|S|\le K}\sum_{x\in S}(f_{x,1}+val_{k,x}-f_{x,0})\\f_{k,1}=\sum_{x\in son_{k}}f_{x,0}+\max_{S\subseteq son_{k},|S|<K}\sum_{x\in S}(f_{x,1}+val_{k,x}-f_{x,0})\end{cases}
$$
(其中$son_{k}$为$k$儿子的集合,$val_{x,y}$表示边$(x,y)$的边权)

对于后者,我们可以维护一个数据结构,支持加入一个元素和查询最大的$K$(或$K-1$)个元素之和

可以使用平衡树/权值线段树实现,单次复杂度为$o(\log n)$,总复杂度为$o(n\log n)$

记$deg_{k}$为节点$k$的度数,将所有边$(x,y)$分为三类:

1.$deg_{x},deg_{y}\le K$,这一类边一定可以选择,直接将$val_{(x,y)}$加入答案并删除

2.$deg_{x}\le K<deg_{y}$,这一类边实际上仅在$y$上有限制,我们可以在求$f_{y,0/1}$时的平衡树中先加入此边权即可

另外,为了维护,在dp结束后要删除$f_{x,1}+val_{k,x}-f_{x,0}$,如果直接在multiset中删除会导致节点个数

3.$deg_{x},deg_{y}>K$,这一类边用之前的dp即可

由此,每一次dp的节点数只有$deg_{k}>K$的节点,而$\sum_{K=0}^{n-1}\sum_{deg_{k}> K}1=o(n)$,总复杂度即$o(n\log n)$

(另外注意搜索时,$deg_{k}>K$的点的出边中要删除$deg_{k}\le K$的点,来保证复杂度)

 1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 100005
4 #define ll long long
5 #define pii pair<int,int>
6 #define mid (l+r>>1)
7 struct Edge{
8 int nex,to,len;
9 }edge[N<<1];
10 vector<pii>G[N];
11 vector<ll>ans,v[N];
12 int E,V,n,tmp,r[N],id[N],head[N],work[N],vis[N],rt[N],ls[N*100],rs[N*100],sz[N*100];
13 ll sum,f[N*100],dp[N][2];
14 bool cmp1(int x,int y){
15 return r[x]<r[y];
16 }
17 bool cmp2(pii x,pii y){
18 return r[x.first]>r[y.first];
19 }
20 void update(int &k,int l,int r,int x,int y){
21 if (!k)k=++V;
22 sz[k]+=y,f[k]+=x*y;
23 if (l==r)return;
24 if (x<=mid)update(ls[k],l,mid,x,y);
25 else update(rs[k],mid+1,r,x,y);
26 }
27 ll query(int k,int l,int r,int x){
28 if (l==r)return 1LL*min(x,sz[k])*l;
29 if (sz[rs[k]]>=x)return query(rs[k],mid+1,r,x);
30 return f[rs[k]]+query(ls[k],l,mid,x-sz[rs[k]]);
31 }
32 void add(int x,int y,int z){
33 edge[E].nex=head[x];
34 edge[E].to=y;
35 edge[E].len=z;
36 head[x]=E++;
37 }
38 void dfs(int k){
39 vis[k]=1;
40 dp[k][0]=dp[k][1]=0;
41 for(int &i=work[k];i!=-1;i=edge[i].nex)
42 if (r[edge[i].to]>tmp)break;
43 for(int i=work[k];i!=-1;i=edge[i].nex)
44 if (!vis[edge[i].to]){
45 dfs(edge[i].to);
46 dp[k][0]+=dp[edge[i].to][0];
47 ll s=dp[edge[i].to][1]+edge[i].len-dp[edge[i].to][0];
48 if (s>0){
49 update(rt[k],1,1e9,s,1);
50 v[k].push_back(s);
51 }
52 }
53 dp[k][1]=dp[k][0];
54 dp[k][0]+=query(rt[k],1,1e9,tmp);
55 dp[k][1]+=query(rt[k],1,1e9,tmp-1);
56 for(int i=0;i<v[k].size();i++)update(rt[k],1,1e9,v[k][i],-1);
57 v[k].clear();
58 }
59 vector<ll> minimum_closure_costs(int n,vector<int>u,vector<int>v,vector<int>w){
60 memset(head,-1,sizeof(head));
61 for(int i=0;i<=n-2;i++){
62 u[i]++,v[i]++;
63 G[u[i]].push_back(make_pair(v[i],w[i]));
64 G[v[i]].push_back(make_pair(u[i],w[i]));
65 sum+=w[i];
66 }
67 for(int i=1;i<=n;i++){
68 id[i]=i;
69 r[i]=G[i].size();
70 }
71 sort(id+1,id+n+1,cmp1);
72 for(int i=1;i<=n;i++){
73 sort(G[i].begin(),G[i].end(),cmp2);
74 for(int j=0;j<G[i].size();j++)add(i,G[i][j].first,G[i][j].second);
75 }
76 memcpy(work,head,sizeof(work));
77 ans.push_back(sum);
78 for(int i=1,j=1;i<n;i++){
79 while ((j<=n)&&(r[id[j]]<=i)){
80 for(int k=head[id[j]];k!=-1;k=edge[k].nex){
81 if (vis[edge[k].to])sum-=edge[k].len;
82 else update(rt[edge[k].to],1,1e9,edge[k].len,1);
83 }
84 vis[id[j++]]=1;
85 }
86 tmp=i;
87 ans.push_back(sum);
88 for(int k=j;k<=n;k++)
89 if (!vis[id[k]]){
90 dfs(id[k]);
91 ans[i]-=dp[id[k]][0];
92 }
93 for(int k=j;k<=n;k++)vis[id[k]]=0;
94 }
95 return ans;
96 }

[luogu7600]封闭道路的更多相关文章

  1. P7600-[APIO2021]封闭道路【堆,dp】

    正题 题目链接:https://www.luogu.com.cn/problem/P7600 题目大意 给出\(n\)个点的一棵树,边有边权,对于每个\(k\)求去掉最小边权和的点使得每个点的度数都不 ...

  2. Dalvik虚拟机Java堆创建过程分析

    文章转载至罗升阳CSDN社区博客,原地址: http://blog.csdn.net/luoshengyang/article/details/6557518 近年来,手机移动平台越来越火爆.打开自己 ...

  3. Apollo框架试玩

    2017年7月5日,百度举行了AI开发者大会,在会上发布了Apollo项目,并进行了演示,该项目在Github上已经能够被访问.出于一个程序员的好奇,昨天试玩了一把,确实不错. http://apol ...

  4. Apollo自动驾驶框架试玩

    2017年7月5日,百度举行了AI开发者大会,在会上发布了Apollo项目,并进行了演示,该项目在Github上已经能够被访问.出于一个程序员的好奇,昨天试玩了一把,确实不错. http://apol ...

  5. Google Waymo自动驾驶安全技术报告(二)

    Waymo的技术在公开道路上.封闭测试场.仿真器进行了广泛的测试,所以可以保证自动驾驶系统的每一部分在其ODD内都有强大.可靠.安全的处理能力. Waymo的自动驾驶系统由三个相互独立.严格测试的子系 ...

  6. AI在出行场景的应用实践:路线规划、ETA、动态事件挖掘…

    ​前言:又到春招季!作为国民级出行服务平台,高德业务快速发展,大量校招/社招名额开放,欢迎大家投递简历,详情见文末.为帮助大家更了解高德技术,我们策划了#春招专栏#的系列文章,组织各业务团队的高年级同 ...

  7. “医疗信息化行业之中的联发科”- 我们在医疗行业中的定位及目标 想做一个面对中小企业的专业上游软件供应商 台湾联发科技颠覆掉的是一个封闭的手机产业系统 解决方案,即AgileHIS.NET数字化医院基础方案

    “医疗信息化行业之中的联发科”- 我们在医疗行业中的定位及目标   我们做中国医疗信息化行业之中的联发科 ---我们在医疗行业中的定位及目标 从我个人来讲,我从2001年到现在这10年之间基本上一直在 ...

  8. 自动驾驶汽车数据不再封闭,Uber 开源新的数据可视化系统

    日前,Uber 开源了基于 web 的自动驾驶可视化系统(AVS),称该系统为自动驾驶行业带来理解和共享数据的新方式.AVS 由Uber旗下负责自动驾驶汽车研发的技术事业群(ATG)开发,目前该系统已 ...

  9. 免费道路 bzoj 3624

    免费道路(1s 128MB)roads [输入样例] 5 7 21 3 04 5 13 2 05 3 14 3 01 2 14 2 1 [输出样例] 3 2 04 3 05 3 11 2 1 题解: ...

随机推荐

  1. 项目实战 Prometheus环境搭建

    项目摘要: 本文是搭建一套prometheus环境的教程. 前期准备:准备三台虚拟机,本文以centos7为例. 项目具体实施:分别进入每台虚拟机设置hostname:# hostnamectl se ...

  2. 创建HTML文档

    目录 创建HTML文档 构筑基本的文档结构 DOCTYPE元素 DOCTYPE元素 代码清单1 使用DOCTYPE元素 html元素 html元素 代码清单2 使用html元素 head元素 head ...

  3. PAT (Basic Level) Practice (中文)1007 素数对猜想 (20分)

    1007 素数对猜想 (20分) 让我们定义d​n为:dn = pn+1 − pn,其中p​i是第i个素数.显然有d1 = 1,且对于n > 1有dn是偶数."素数对猜想"认 ...

  4. Golang通脉之数组

    数组是同一种数据类型元素的集合.数组在内存中都是连续存放的. 在Go语言中,数组从声明时就确定,使用时可以修改数组成员,但是数组大小不可变化. 基本语法: // 定义一个长度为3元素类型为int的数组 ...

  5. Java---String和StringBuffer类

    Java---String和StringBuffer类 Java String 类 字符串在Java中属于对象,Java提供String类来创建和操作字符串. 创建字符串 创建字符串常用的方法如下: ...

  6. HCIP-RSTP

    端口角色 根端口(RP): 非根桥收到最优的BPDU配置信息的端口为根端口,(到根桥开销最小的端口),根桥没有根端口. 指定端口(DP): 两非根交换机之间连接的两个端口必有一个为指定端口,比较两个非 ...

  7. Java多线程中的死锁

    Java多线程中的死锁 死锁产生的原因 线程死锁是指由两个以上的线程互相持有对方所需要的资源,导致线程处于等待状态,无法往前执行. 当线程进入对象的synchronized代码块时,便占有了资源,直到 ...

  8. Java:并发笔记-08

    Java:并发笔记-08 说明:这是看了 bilibili 上 黑马程序员 的课程 java并发编程 后做的笔记 7. 共享模型之工具-1 7.1 线程池 1. 自定义线程池 步骤1:自定义拒绝策略接 ...

  9. Alpha阶段初始任务分配

    项目 内容 这个作业属于哪个课程 2021春季软件工程(罗杰 任健) 这个作业的要求在哪里 团队项目-计划-Alpha阶段说明书 一.Alpha阶段总体规划 进行服务器相关部署 进行开发相关技术学习 ...

  10. [no code][scrum meeting] Alpha 4

    项目 内容 会议时间 2020-04-09 会议主题 OCR相关的技术展示与讨论 会议时长 30min 参会人员 全体成员 $( "#cnblogs_post_body" ).ca ...