长剖与贪心+树上反悔贪心:1004T4

news2024/11/18 23:38:37

长剖的本质是一种贪心。(启发式合并本质也是类似哈夫曼树的过程)

在此题中,首先肯定变直径,然后选端点为根。然后选叶子。而每个叶子为了不重复计算,可以只计算其长剖后所在链的贡献。(本题精髓,用长剖来贪心)

然后钦定某个点必选,就是一种反悔贪心。很显然的思路是删掉排名 2 ∗ k − 1 2*k-1 2k1 的叶子,但考虑:

在这里插入图片描述

所以需要考虑离其最近被选的点


#include<bits/stdc++.h>
using namespace std;
//#define int long long
inline int read(){int x=0,f=1;char ch=getchar(); while(ch<'0'||
ch>'9'){if(ch=='-')f=-1;ch=getchar();}while(ch>='0'&&ch<='9'){
x=(x<<1)+(x<<3)+(ch^48);ch=getchar();}return x*f;}
#define Z(x) (x)*(x)
#define pb push_back
//mt19937 rand(time(0));
//mt19937_64 rand(time(0));
//srand(time(0));
#define N 500010
//#define M
//#define mo
struct node { int x; long long y, z; };
int n, m, i, j, k, T, p1, p2, in[N];
int u, v, w, qe; 
vector<node>G[N]; 

struct Tree {
	int i, j, k, rt, mn[N]; 
	long long h[N], mxh[N], mx[N], sum[N]; 
	int son[N], dep[N], top[N]; 
	int f[N][22], rk[N], dfn[N]; 
	node w[N]; 
	void dfs1(int x, int fa, int &p1) {//p1 p2
		if(h[x]>h[p1]) p1=x;  
		for(auto t : G[x]) {
			int y=t.y; 
			long long z=t.z; 
			if(y==fa) continue; 
			h[y]=h[x]+z; dfs1(y, x, p1); 
		}
	}
	void dfs2(int x, int fa) { //son[x] h[x] dep[x]
		dep[x]=dep[fa]+1; mx[x]=mxh[x]=h[x]; 
		for(auto t : G[x]) {
			int y=t.y; 
			long long z=t.z; 
			if(y==fa) continue; 
			h[y]=h[x]+z; 
//			printf("%lld(%lld) --%lld-> %lld(%lld)\n", x, h[x], z, y, h[y]); 

			dfs2(y, x); mx[x]=max(mx[x], mx[y]); 
			if(mxh[y]>mxh[son[x]]) son[x]=y; 
		}
		if(son[x]) mxh[x]=mxh[son[x]]; 
	}
	void dfs3(int x, int fa, int tp) {//top[x] w[x] 
//		printf("> %d\n", tp); 
		top[x]=tp; f[x][0]=fa; 
		if(in[x]==1 && fa) {
			w[x].y=h[x]-h[f[top[x]][0]]; 
			w[x].x=x; 
		}
		for(auto t : G[x]) {
			int y=t.y; 			
			if(y==fa) continue; 
			if(y==son[x]) dfs3(y, x, tp); 
			else dfs3(y, x, y); 
		}
	}
	void init() {
//		for(i=1; i<=n; ++i) printf("%d ", top[i]); printf("\n"); 
//		for(i=1; i<=n; ++i) printf("%d ", h[i]); printf("\n"); 
		sort(w+1, w+n+1, [] (node x, node y) { return x.y<y.y; }) ; 
		reverse(w+1, w+n+1); 
		for(i=1; i<=n; ++i) {
//			printf("%lld(%lld) ", w[i].y, w[i].x); 
			if(w[i].x) sum[i]=w[i].y, rk[w[i].x]=i, dfn[i]=w[i].x; 
			sum[i]+=sum[i-1]; 
		}
//		printf("\n"); 
		for(k=1; k<=19; ++k) 
			for(i=1; i<=n; ++i) f[i][k]=f[f[i][k-1]][k-1]; 
	}
	void dfs4(int x, int fa) {
		if(in[x]==1 && fa) mn[x]=rk[x]; else mn[x]=1e9; 
		for(auto t : G[x]) {
			int y=t.y, z=t.z; 
			if(y==fa) continue; 
			dfs4(y, x); mn[x]=min(mn[x], mn[y]); //排名最小 
		}
	}
	int tiao(int x, int g) {
		for(k=19; k>=0; --k)
		 	if(mn[f[x][k]]>g) x=f[x][k]; 
		return f[x][0]; 
	}
	int lca(int x, int y) {
		if(x==y) return x; 
		if(dep[x]<dep[y]) swap(x, y); 
		for(int k=19; k>=0; --k)
		 	if(dep[f[x][k]]>=dep[y]) x=f[x][k]; 
		if(x==y) return x; 
		for(int k=19; k>=0; --k)
		 	if(f[x][k]!=f[y][k]) x=f[x][k], y=f[y][k]; 
		return f[x][0]; 
	}
	long long calc(int y, int oldy, int newx) {
//		printf("Lca(%d %d) : %d\n", oldy, newx, lca(oldy, newx)); 
//		return min(w[mn[y]].y, h[oldy]-h[lca(oldy, newx)]); 
		return min(w[mn[y]].y, h[oldy]-h[y]); 
	}
	long long que(int x, int k) {
		if(k==1) {
//			int y=dfn[mn[x]]; return h[y]; 
			return mx[x]; 
		}
		if(mn[x]<=2*k-1) {
			return sum[min(2*k-1, n)]; 
		}
		int y=tiao(x, 2*k-1), newx, oldy; 
		long long ans; 
		newx=dfn[mn[x]]; oldy=dfn[mn[y]]; 
//		printf("%d | %d %d %d %d\n", y, newx, oldy, (h[newx]-h[y]), calc(y, oldy, newx)); 

		ans=sum[2*k-1]-calc(y, oldy, newx)+(h[newx]-h[y]); 
		ans=max(ans, sum[2*k-1]-w[2*k-1].y+(h[newx]-h[y])); 
		return ans; 
	}
}T1, T2;

void print(long long x) {
	if(x) print(x/10), putchar(x%10+'0'); 
}

signed main()
{
//	freopen("in.txt", "r", stdin);
//	freopen("out.txt", "w", stdout);
		freopen("bomb.in", "r", stdin);
	freopen("bomb.out", "w", stdout);
//	T=read();
//	while(T--) {
//
//	}
	n=read(); qe=read(); 
	for(i=1; i<n; ++i) {
		u=read(); v=read(); w=read(); 
		G[u].pb({u, v, w}); 
		G[v].pb({v, u, w}); 
		++in[u]; ++in[v]; 
	}
	T1.h[1]=0;  T1.dfs1(1, 0, p1); 
	T1.h[p1]=0; T1.dfs1(p1, 0, p2);
	T1.rt=p1; T2.rt=p2; 
	T1.h[p1]=0; T1.dfs2(p1, 0); 
	T2.h[p2]=0; T2.dfs2(p2, 0); 
//	printf("%d %d\n", p1, p2); 
	T1.dfs3(p1, 0, p1); 
	T2.dfs3(p2, 0, p2); 
	T1.init(); T2.init(); 
	T1.dfs4(p1, 0); T2.dfs4(p2, 0); 
	
	while(qe--) {
		u=read(); k=read(); 
		print(max(T1.que(u, k), T2.que(u, k))); puts(""); 
	}
	return 0;
}


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1062728.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Postman使用实例

Postman使用实例 实体类Emp package com.example.springboot_postman.pojo;import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor;import javax.persistence.*; import j…

【C语言】善于利用指针(一)

&#x1f497;个人主页&#x1f497; ⭐个人专栏——C语言初步学习⭐ &#x1f4ab;点击关注&#x1f929;一起学习C语言&#x1f4af;&#x1f4ab; 目录 导读&#xff1a; 1. 什么是指针 1.1 概念 1.2 图解 1.3 示例 2. 指针和指针类型 2.1 指针的定义 2.2 指针的解引…

Kubernetes Deployment的运行

Deployment提供了一种更加简单的更新Replication Controller和Pod的机制&#xff0c;更好地解决了Pod的编排问题。本节将详细介绍如何通过Deployment实现Pod的管理。 15.1.1 什么是Deployment Deployment的中文意思为部署、调集&#xff0c;它是在Kubernetes的1.2版本中新增…

【kubernetes】基于prometheus的监控

目录 1 监控解决方案2 prometheus2.1 容器监控2.2 节点监控2.3 资源对象监控2.4 metrics--server 3 prometheus-operator vs kube-prometheus vs helm3.1 prometheus-operator3.2 kube-prometheus3.3 helm 参考文档 1 监控解决方案 从实现方案来说&#xff0c;监控分为3个部分…

计算机网络基础(二):物理层、数据链路层及网络层

一、物理层 1.物理层 物理层面的通信标准可以概括划分为与网络基础设施有关的标准和与被传输物理信号有关的标准两类。 网络基础设施的标准&#xff1a;鉴于物理层面的消息互通也是物理层应该兑现的服务&#xff0c;因此物理层的标准还会包括针脚的用途、线缆的材料与设计等…

vertx的学习总结7之用kotlin 与vertx搞一个简单的http

这里我就简单的聊几句&#xff0c;如何用vertx web来搞一个web项目的 1、首先先引入几个依赖&#xff0c;这里我就用maven了&#xff0c;这个是kotlinvertx web <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apac…

华为云云耀云服务器L实例评测|基于canal缓存自动更新流程 SpringBoot项目应用案例和源码

前言 最近华为云云耀云服务器L实例上新&#xff0c;也搞了一台来玩&#xff0c;期间遇到各种问题&#xff0c;在解决问题的过程中学到不少和运维相关的知识。 在之前的博客中&#xff0c;介绍过canal的安装和配置&#xff0c;参考博客 拉取创建canal镜像配置相关参数 & …

【数据库——MySQL】(14)过程式对象程序设计——游标、触发器

目录 1. 游标1.1 声明游标1.2 打开游标1.3 读取游标1.4 关闭游标1.5 游标示例 2. 触发器2.1 创建触发器2.2 修改触发器2.3 删除触发器2.4 触发器类型2.5 触发器示例 参考书籍 1. 游标 游标一般和存储过程一起配合使用。 1.1 声明游标 要使用游标&#xff0c;需要用到 DECLAR…

<C++> String

目录 一、标准库中的string类 1. string类 2. string类的常用接口说明 2.1 string类对象的常见构造 2.2 string类对象的容量操作 2.3 string类对象的访问及遍历操作 2.4 string类对象的修改操作 2.5 string类非成员函数 总结 前言 C语言中&#xff0c;字符串是以 \0 结尾的一些…

用JMeter对HTTP接口进行压测(一)压测脚本的书写、调试思路

文章目录 安装JMeter和Groovy为什么选择Groovy&#xff1f; 压测需求以及思路准备JMeter脚本以及脚本正确性验证使用Test Script Recorder来获取整条业务线上涉及的接口为什么使用Test Script Recorder&#xff1f; 配置Test Script Recorder对接口进行动态化处理处理全局变量以…

2. 资源管理

2. 资源管理 文章目录 2. 资源管理2.1 资源管理介绍2.2 YAML语言介绍2.3 资源管理方式2.2.1 命令式对象管理2.2.2 命令式对象配置2.2.3 声明式对象配置 2.4. 模拟使用普通用户来操作 2.1 资源管理介绍 在kubernetes中&#xff0c;所有的内容都抽象为资源&#xff0c;用户需要通…

二十九、【进阶】MySQL索引的概述和索引查询

1、索引概述 2、 普通查询和索引查询 &#xff08;1&#xff09;基础演示 无索引查询&#xff1a;在查询信息时&#xff0c;比如查询年龄age45的员工&#xff0c;系统会遍历字段为age的列&#xff0c;在找到age45的员工后&#xff0c;依旧会向下扫描&#xff0c;直到表末&…

如何使用 Dijkstra 算法找到从源到所有顶点的最短路径--附C++/Java源码

给定一个图和图中的源顶点,找到从源到给定图中所有顶点的最短路径。 例子: 输入: src = 0,图形如下图所示。 输出: 0 4 12 19 21 11 9 8 14解释:从 0 到 1 的距离 = 4。 从 0 到 2 的最小距离 = 12。0->1->2 从 0 到 3 的最小距离 = 19。0 ->1-

Python基础语法(3)

目录 一、函数 1.1 函数是什么 1.2 函数语法格式 1.3 函数参数 1.4 函数返回值 a. 一个函数中可以有多个 return 语句 b. 执行到 return 语句&#xff0c;函数就会立即执行结束&#xff0c;回到调用位置 c. 一个函数是可以一次返回多个返回值的。使用 , 来分割多个返回值…

Jmeter基础篇

1.性能测试指标 【虚拟用户数】&#xff1a;线程用户 【并发数】&#xff1a;指在某一时间&#xff0c;一定数量的虚拟用户同时对系统的某个功能进行交互&#xff0c;一般通过集合点实现 【事务】:事务代表一个完整的功能&#xff0c;一个接口可以是事务&#xff0c;多个接口…

MyBatisPlus(十一)包含查询:in

说明 包含查询&#xff0c;对应SQL语句中的 in 语句&#xff0c;查询参数包含在入参列表之内的数据。 in Testvoid inNonEmptyList() {// 非空列表&#xff0c;作为参数List<Integer> ages Stream.of(18, 20, 22).collect(Collectors.toList());in(ages);}Testvoid in…

【C语言】转圈报数问题(三种方法--指针,数组)

题目&#xff1a;有n个人围成一圈&#xff0c;顺序排号。从第一个人开始报数&#xff08;从1到3报数&#xff09;&#xff0c;凡报到3的人退出圈子&#xff0c;问最后留下的是原来第几号的那位。 方法一&#xff1a; #include <stdio.h> #define N 10int main() {int …

systemverilog function的一点小case

关于function的应用无论是在systemverilog还是verilog中都有很广泛的应用&#xff0c;但是一直有一个模糊的概念困扰着我&#xff0c;今天刚好有时间来搞清楚并记录下来。 关于fucntion的返回值的问题&#xff1a; function integer clog2( input logic[255:0] value);for(cl…

实验三十五、LM117 稳压电源的设计

一、题目 利用 LM117 设计一个稳压电路&#xff0c;要求输出电压的调节范围为 5 ∼ 20 V 5\sim20\,\textrm V 5∼20V&#xff0c;最大负载电流为 400 mA 400\,\textrm{mA} 400mA。利用 Multisim 对所设计电路进行仿真&#xff0c;并测试所有性能指标。 二、仿真电路 仿真电…

安装NodeJS并使用yarn下载前端依赖

文章目录 1、安装NodeJS1.1 下载NodeJS安装包1.2 解压并配置NodeJS1.3 验证是否安装成功2、使用yarn下载前端依赖2.1 安装yarn2.2 使用yarn下载前端依赖参考目标:在Windows下安装新版NodeJS,并使用yarn下载前端依赖,实现运行前端项目。 1、安装NodeJS 1.1 下载NodeJS安装包…