动态树的最值

news2025/3/2 2:37:18

一 问题描述

一棵树有 N 个节点,每个节点都有一个权值 Wi ,有 4 种操作。

① 1 x y ,在两个节点 x、y 之间添加一条新边。因此,在这种操作之后,两棵树将连接成一棵新树。

② 2 x y ,在树集合中找到包含节点 x 的树,并且使节点 x 成为该树的根,然后删除节点 y 与其父节点之间的边。在这种操作之后,一棵树将被分成两部分。

③ 3 w x y ,将 x 到 y 路径上所有节点的权值都增加w 。

④ 4 x y ,查询 x 到 y 路径上节点的最大权值。

二 输入和输出

1 输入

包含多个测试用例。每个测试用例的第 1 行都包含一个整数 N(1 ≤ N ≤3×10^5 );接下来的 N -1 行,每行都包含两个整数 x、y ,表示它们之间有一条边;下一行包含 N 个整数,表示每个节点的权值都为 Wi(0≤W i ≤3000);再下一行包含一个整数Q(1≤Q≤3×10^5 ),接下来的 Q 行以整数 1、2、3 或 4 为开头,表示操作类型。

2 输出

对每个查询,都单行输出正确答案。若此查询是特殊操作,则输出 -1。在每个测试用例之后都输出一个空行。

三 输入和输出样例

1 输入样例

5

1 2

2 4

2 5

1 3

1 2 3 4 5

6

4 2 3

2 1 2

4 2 3

1 3 5

3 2 1 4

4 1 4

2 输出样例

3

-1

7

四 注意

在第 1 种操作中,若节点 x 、y 属于同一棵树,则是特殊的。

在第 2 种操作中,若x =y 或者x 、y 不属于同一棵树,则是特殊的。

在第 3 种操作中,若x 、y 不属于同一棵树,则是特殊的。

在第 4 种操作中,若x 、y 不属于同一棵树,则是特殊的。

五 分析

根据输入样例,构建的树形结构如下图所示。

输入样例的 6 种操作如下。

(1)4 2 3:查询 2-3 的最大节点权值,输出 3。

(2)2 1 2:找到包含节点 1 的树,使节点 1 成为该树的根,然后删除节点 2 与其父节点之间的边,实际上就是删除 1-2 的边。

(3)4 2 3:查询 2-3 的最大节点权值,2、3 不属于同一棵树,是非法的,输出 -1。

(4)1 3 5:连接 3-5 的边。

(5)3 2 1 4:将 1-4 路径上的节点权值加 2。

(6)4 1 4:查询 1-4 的最大节点权值,输出 7。

六 设计

本问题包含 4 种基本操作。

(1)连边:在 x、y 之间连接一条新边,若 x、y 属于同一棵树,则非法,输出 -1;否则执行 link(x , y)。

(2)删边:删除 x、y 之间的边。若 x=y 或者 x、y 不属于同一棵树,则非法,输出 -1;否则执行 cut(x , y )。

(3)区间更新:将 x 到 y 路径上所有节点的权值都增加 w 。若节点 x、y 不属于同一棵树,则非法,输出-1;否则执行 addval(x , y ,w )。

(4)区间最值查询:输出 x 到 y 路径上节点的最大权值。若节点 x、y 不属于同一棵树,则非法,输出-1;否则执行 split(x , y ),输出 mx[y]。

七 代码

package com.platform.modules.alg.alglib.hdu4010;

public class Hdu4010 {
    private int inf = 0x3f3f3f3f;
    private int N = 300005;

    int n, m, top;
    int c[][] = new int[N][2];
    int fa[] = new int[N];
    int v[] = new int[N];
    int st[] = new int[N];
    int add[] = new int[N];
    int mx[] = new int[N];
    int rev[] = new int[N];

    void update(int x) {
        int l = c[x][0], r = c[x][1];
        mx[x] = Math.max(mx[l], mx[r]);
        mx[x] = Math.max(mx[x], v[x]);
    }

    void pushdown(int x) {
        int l = c[x][0], r = c[x][1];
        if (rev[x] == 1) {
            rev[l] ^= 1;
            rev[r] ^= 1;
            rev[x] ^= 1;
            int temp = c[x][0];
            c[x][0] = c[x][1];
            c[x][1] = temp;
        }
        if (add[x] > 0) {
            if (l > 0) {
                add[l] += add[x];
                mx[l] += add[x];
                v[l] += add[x];
            }
            if (r > 0) {
                add[r] += add[x];
                mx[r] += add[x];
                v[r] += add[x];
            }
            add[x] = 0;
        }
    }

    boolean isroot(int x) {
        return c[fa[x]][0] != x && c[fa[x]][1] != x;
    }

    void rotate(int x) {
        int y = fa[x], z = fa[y], k;
        if (c[y][0] == x) {
            k = 1;
        } else {
            k = 0;
        }
        if (!isroot(y)) c[z][c[z][1] == y ? 1 : 0] = x;
        fa[x] = z;
        fa[y] = x;
        fa[c[x][k]] = y;
        c[y][k == 0 ? 1 : 0] = c[x][k];
        c[x][k] = y;
        update(y);
        update(x);
    }

    void splay(int x) {
        top = 0;
        st[++top] = x;
        for (int i = x; !isroot(i); i = fa[i])
            st[++top] = fa[i];
        while (top > 0) pushdown(st[top--]);
        while (!isroot(x)) {
            int y = fa[x], z = fa[y];
            if (!isroot(y)) {
                if (c[y][0] == x ^ c[z][0] == y) {
                    rotate(x);
                } else {
                    rotate(y);
                }
            }
            rotate(x);
        }
    }

    void access(int x) {
        for (int t = 0; x > 0; t = x, x = fa[x]) {
            splay(x);
            c[x][1] = t;
            update(x);
        }
    }

    void makeroot(int x) {
        access(x);
        splay(x);
        rev[x] ^= 1;
    }

    void link(int x, int y) {
        makeroot(x);
        fa[x] = y;
    }

    void split(int x, int y) {
        makeroot(x);
        access(y);
        splay(y);
    }

    void cut(int x, int y) {
        split(x, y);
        c[y][0] = fa[c[y][0]] = 0;
        update(y);
    }

    int findroot(int x) {
        access(x);
        splay(x);
        while (c[x][0] > 0) x = c[x][0];
        return x;
    }

    void addval(int x, int y, int val) {
        split(x, y);
        add[y] += val;
        mx[y] += val;
        v[y] += val;
    }

    public String output = "";

    public String cal(String input) {
        int opt, x, y, w;
        String[] line = input.split("\n");
        n = Integer.parseInt(line[0]);

        for (int i = 0; i <= n; i++)
            add[i] = rev[i] = fa[i] = c[i][0] = c[i][1] = 0;
        mx[0] = -inf;
        for (int i = 1; i < n; i++) {
            String[] num = line[i].split(" ");
            x = Integer.parseInt(num[0]);
            y = Integer.parseInt(num[1]);
            link(x, y);
        }
        String[] wi = line[n].split(" ");
        for (int i = 1; i <= n; i++) {
            v[i] = Integer.parseInt(wi[i - 1]);
            mx[i] = v[i];
        }
        m = Integer.parseInt(line[n + 1]);
        int count = 0;
        while (m-- > 0) {
            String[] query = line[n + 2 + count++].split(" ");
            opt = Integer.parseInt(query[0]);
            x = Integer.parseInt(query[1]);
            y = Integer.parseInt(query[2]);
            switch (opt) {
                case 1:
                    if (findroot(x) == findroot(y)) {
                        output += "-1\n";
                        break;
                    }
                    link(x, y);
                    break;
                case 2:
                    if (findroot(x) != findroot(y) || x == y) {
                        output += "-1\n";
                        break;
                    }
                    cut(x, y);
                    break;
                case 3:
                    w = x;
                    x = y;
                    y = Integer.parseInt(query[3]);
                    if (findroot(x) != findroot(y)) {
                        output += "-1\n";
                        break;
                    }
                    addval(x, y, w);
                    break;
                case 4:
                    if (findroot(x) != findroot(y)) {
                        output += "-1\n";
                        break;
                    }
                    split(x, y);
                    output += mx[y] + "\n";
                    break;
            }
        }
        return output;
    }
}

八 测试

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

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

相关文章

LeetCOde-剑指46-把数字翻译成字符串

1、动态规划法 我们通过观察可以发现&#xff0c;假如我们使用数组dp[i]dp[i]dp[i]来记录前iii位可能构成的字符串个数&#xff1a;1、当新加入的第i1i1i1位和第iii位能够构成一个大于9小于26的数字时&#xff0c;dp[i1]dp[i]dp[i−1]dp[i1]dp[i]dp[i-1]dp[i1]dp[i]dp[i−1]&a…

容器化部署(k8s)任务调度平台xxl-job(部署过程及踩坑问题记录)

文章预览&#xff1a;1 部署过程&#xff08;下方ip代表服务器的ip哈&#xff09;1.1 制作服务打包镜像DockerFile1.2 制作执行脚本run.sh1.3 jar包上上传1.4 kuboard创建----配置信息2 踩坑问题记录2.1 日志抛出异常2.2 原因分析2.3 过程分析及解决2.4 执行调度测试导入sql等过…

Baklib|SaaS产品,实现企业流程数字化

正如许多科技潮流一样&#xff0c;“SaaS”这个词也逐渐成为企业经理们谈论的话题。然而&#xff0c;如果您对“SaaS”一无所知&#xff0c;您可能会感到困惑并容易忽略它。那么&#xff0c;什么是“SaaS”&#xff1f;它的优点是什么&#xff1f;它如何帮助企业实现数字化转型…

SSM+VUE+ElementUI实现宠物领养系统,期末大作业

SSMVUEElementUI实现宠物领养系统 系统角色 领养人&#xff0c;管理员 系统功能 本系统的功能主要分为四大模块&#xff1a; 领养人用户模块&#xff1a;注册、领养人登录、申请领养、查看小动物信息、发布留言领养机构员工用户模块&#xff1a;领养机构员工登录、增加小动…

地平线开发者社区真心话大冒险,邀你闯关!

Hello&#xff01; 各位初次见面的萌新和久经沙场的社牛 目前开发者社区已成立两年有余 感谢大家一路上的支持和理解 今天&#xff0c;我们也准备了一些小礼品 希望倾听大家作为用户和开发者的真心话 同时&#xff0c;也欢迎初次见面的萌新们一同冒险 期待陪伴大家走过更…

Protect Privacy from Gradient Leakage Attack in Federated Learning

wangjunxiao/GradDefense: Defense against Gradient Leakage Attack (github.com) Summary 针对DGA和DIA攻击&#xff0c;提出了一个轻量、保证训练准确性、够用的的防御机制。防御机制主要包括随机layer添加扰动&#xff0c;然后进行梯度补偿来减少噪声对模型准确性的影响。…

CORS处理跨域问题

“前后端分离的项目必然会遇到一个典型的问题——跨域问题。” 跨域 要解决跨域问题&#xff0c;首先得知道什么是跨域&#xff1f; 首先&#xff0c;跨域是访问的域名或IP、端口三者有一不同都属于跨域。&#xff08;注意请求路径不是&#xff09;&#xff0c;即使在本地测试&…

【计算机网络】学习笔记--第一章

【计算机网络】学习笔记--第一章基本概念端系统之间的通信客户-服务器方式&#xff08;C/S方式&#xff09;对等连接方式&#xff1a;三种交换方式电路交换&#xff08;Circuit Switching&#xff09;分组交换&#xff08;Packet Switching&#xff09;报文交换&#xff08;Mes…

centos7 安装与卸载 Mysql 5.7.27(详细完整教程)

目录 卸载 安装 卸载 1、关闭MySQL服务 systemctl stop mysqld2、使用 rpm 命令查看已安装的安装包 [nameVM-20-12-centos mysql1]$ rpm -qa|grep mysql 3、使用yum卸载安装的mysql [nameVM-20-12-centos mysql1]$ sudo yum remove mysql mysql-server mysql-libs mysql…

【LeetCode每日一题:775.全局倒置与局部倒置~~~维护前缀最大值】

题目描述 给你一个长度为 n 的整数数组 nums &#xff0c;表示由范围 [0, n - 1] 内所有整数组成的一个排列。 全局倒置 的数目等于满足下述条件不同下标对 (i, j) 的数目&#xff1a; 0 < i < j < n nums[i] > nums[j] 局部倒置 的数目等于满足下述条件的下标 …

Air780E连接点灯科技-LuatOS

前面发了腾讯云点灯以后&#xff0c;有朋友提到了点灯科技&#xff0c;人家都叫点灯科技了&#xff0c;那咱们学点灯必须连一下试试呀。本文将记录如何将设备连接点灯云平台&#xff0c;并通过手机进行远程控制 先上成果演示 一、点灯云准备 下载点灯科技的app-blinker&#…

Offsets 获取该行的起始索引 start=offsets (x)

获取数据列的第 x 行&#xff0c;通过 Offsets 获取该行的起始索引 startoffsets (x) 和下一行的起始索引 endoffsets (x1)&#xff0c;然后使用这两个索引通过 Bytes 获取具体数据对应的字节流 bytes.slice (start,end)&#xff0c;最后按照字段类型做相应的转换即可。 注&am…

QGC二次开发基础

文章目录 前言一、添加文件到QGC工程二、添加界面三、QML和C++交互四、信号与槽五、测试前言 QGC 4.2.4 一、添加文件到QGC工程 在qgroundcontrol/src目录下创建SimpleTest文件夹 在文件夹中创建SimpleTest.cpp、SimpleTest.h和SimpleTest.qml三个文件 在qgroundcontrol.p…

锁竞争导致的慢sql分析

线上在同步用户时&#xff0c;经常出现简单sql的慢日志。根据方法找到代码&#xff0c;发现方法内使用redisson进行锁操作&#xff0c;waiTime和leaseTime都为3秒,数据库操作比较简单&#xff0c;只是一个简单的用户更新操作。代码简化后如下 Override Transactional(rollback…

客流分析统计摄像头可定制算法程序自动判断识别提醒

客流分析统计摄像头是一个专门为商业零售业企业开发的智能客流量统计分析系统。客流统计分析系统能实时、动态、准确、连续地记录着经营场地的客流的数据信息&#xff0c;既有当前客流又有历史客流&#xff0c;既有不同时段的&#xff0c;又有不同区域客流数据。 客流分析统计摄…

【论文】Poly-yolo: 改进anchor分配问题

文章目录Poly-yolo: higher speed,more precise detection and instance segmentation for yolov31 修改了骨干网络增加CE模块2 重写标签3、修改了输出层3.1 修改细节3.2 修改目的&#xff1a;改进anchor分配问题4 检测多边形 Instance segmentation with Poly-YOLO4.1 The pri…

【Pytorch with fastai】第 9 章 :表格建模深入探讨

&#x1f50e;大家好&#xff0c;我是Sonhhxg_柒&#xff0c;希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流&#x1f50e; &#x1f4dd;个人主页&#xff0d;Sonhhxg_柒的博客_CSDN博客 &#x1f4c3; &#x1f381;欢迎各位→点赞…

95后工程师上班哼小曲?那些愉快上班的打工人,到底怎么做到的?

特别羡慕我们公司一个画PCB板子的刘工&#xff0c;95后&#xff0c;来公司也3年多了&#xff0c;他不是我们部门赚得最多的&#xff0c;也不是人际关系处得最好的&#xff0c;却是活得最开心的。 具体表现在哪呢&#xff1f; ——他居然能每天哼着小曲上班。 怎么会有人上班…

《Linux下的进程创建》

【一】fork函数初识 在Linux中fork函数是非常重要的&#xff0c;他从已存在的进程中创建一个新进程&#xff0c;进程为子进程&#xff0c;而原进程为父进程。 返回值&#xff1a;fork函数的返回值是非常有意思的&#xff0c;他是有两个返回值的&#xff0c;对于父进程来说&…

QRegExp(正则表达式)

QRegExp 头文件&#xff1a;#include<QRegExp> 构造函数&#xff1a; 常用函数&#xff1a; indexIn()判断是否符合规则matchedLength()返回最后一个匹配字符串的长度&#xff0c;没有的话返回-1setPattern()将模式字符串设置为模式。区分大小写、通配符和最小匹配选项不…