【bzoj4712】洪水

Portal --> bzoj4712

Description

   给你一棵树,节点从\(1\)\(n\)编号,每个节点有一个权值,有若干次操作,操作有以下两种:

\((C,x,delta)\):将编号为\(x\)的点的权值改为\(delta\)

\((Q,x)\):询问将\(x\)号节点为根的子树中的所有叶子结点与子树外的其他所有叶子节点分离的最小代价,分离可以通过删除节点实现,删除一个节点的对应代价为该点的权值

   数据范围:\(n<=200000\),任意\(delta\)均为非负数,答案在long long范围内

Solution

   这题貌似是。。经典的动态dp问题,维护重链信息\(f\)和轻儿子信息之和\(g\)之后可以用树剖解决

​   我们先考虑没有修改操作的情况,那就是直接dp

​   记\(f[x]\)表示以\(x\)为根的子树的答案,\(val[x]\)表示\(x\)点的权值,那么有:
\[ f[x]=min(val[x],\sum\limits_{x\rightarrow u}f[u]) \]
​ 如果要修改的话我们考虑树剖,再定义一个数组\(g\)\(g[x]\)表示\(x\)的轻儿子的\(f\)值之和,那么就有:
\[ f[x]=min(val[x],f[son[x]]+g[x]) \]
​   其中\(son[x]\)表示的是\(x\)的重儿子

​   这个时候就有好几种不同的做法了,反正就是维护这两个值就好了,接下来讲的是一种通过维护一个类似转移矩阵一样的东西来用树剖维护的方法

​   我们考虑定义一种新的矩阵乘法(额好像也不能叫做矩阵乘法反正就是一种新的运算):
\[ C[i][j]=min(A[i][k]+B[k][j]) \]
   其中\(A,B,C\)都是矩阵,其实这种运算就是将原来的矩阵乘法的乘法改成了取最小值

   这个运算是满足结合律的,具体证明的话就是我们可以将一次这样的运算理解成求最短路,然后连续两次这样的运算的话就是相当于你有三排点,要求最左边到最右边的最短路,显然你可以先求出左边两排的最短路,或者先求出右边两排的最短路,最后得到的结果是一样的

   定义完这样的运算有什么用呢?我们可以考虑将\(f[x]\)的求值过程用这种运算来实现,为了方便表示我们将这种运算的符号约定为\(*\)
\[ \begin{bmatrix}f[x]\\0\end{bmatrix}=\begin{bmatrix}g[x]&val[x]\\0&0\end{bmatrix}*\begin{bmatrix}f[son[x]]\\0\end{bmatrix} \]
   具体的话就是。。大力展开一下就好了

​   然后我们可以发现\(\begin{bmatrix}f[son[x]]\\0\end{bmatrix}\)是可以写成\(\begin{bmatrix}g[son[son[x]]]&val[son[son[x]]]\\0&0\end{bmatrix}\)的,一路写下去就会发现其实就是\(x\)所在的整条重链中,从\(x\)这个位置的矩阵一路连续运算一直到重链底就好了

​   注意到这个剖完之后就是一段连续的区间,然后这样我们就可以放心用线段树来维护区间的矩阵的连续运算得到的结果,然后其他的就是跟树剖差不多的操作了

​    

​   其实还是有(hen)点(da)区别的,具体一点就是

1、查询

   与普通树剖不同的是这样我们维护的是。。某个位置到链底的权值运算和,所以。。我们需要记录的是每个点所在的链底在哪里,然后查询的时候并不需要往上跳直接查就好了

2、修改

   修改的话因为这里会影响到祖先的\(g\)值,所以我们还是要常规操作一路跳上去,不过不同的是,这里修改直接暴力通过线段树区间查询求得\(f[top[x]\)然后在\(g[pre[top[x]]]\)里面减去(\(pre[x]\)表示\(x\)在原树上的直接祖先,\(top[x]\)表示\(x\)所在的重链头),然后更新线段树中的信息,然后再将新的值加回\(g[pre[top[x]]]\)

   

​   然后就十分愉悦\(O(nlog^2n)\)做完啦,这个做法有一个好处就是。。就算没有保证修改的\(delta\)非负也可以直接做,以及重载运算符之后线段树写起来很爽qwq

​   以及洛谷上面貌似有碾爆LCT和树剖的一个\(log\)做法qwq(orzlyy!!)然而。。我太懒了并没有写qwq

   

​   代码大概长这个样子

#include<iostream>
#include<cstdio>
#include<cstring>
#define ll long long
using namespace std;
const int N=200010,SEG=N*4;
const ll inf=1LL<<60;
struct xxx{int y,nxt;
}a[N*2];
int h[N],lis[N],dfn[N],ed[N],pre[N],sz[N];//ed=the end of the chain
ll f[N],g[N],val[N];
int son[N],top[N];
int n,m,tot,dfn_t;
ll ans;
struct Mtrix{/*{{{*/ll a[2][2];int n;void init(int _n){n=_n;for (int i=0;i<n;++i)for (int j=0;j<n;++j) a[i][j]=inf;}friend Mtrix operator * (Mtrix x,Mtrix y){Mtrix ret; ret.n=x.n;for (int i=0;i<ret.n;++i)for (int j=0;j<ret.n;++j){ret.a[i][j]=inf;for (int k=0;k<ret.n;++k)if (x.a[i][k]!=inf&&y.a[k][j]!=inf)ret.a[i][j]=min(ret.a[i][j],x.a[i][k]+y.a[k][j]);}return ret;}
};/*}}}*/
namespace Seg{/*{{{*/int ch[SEG][2];Mtrix info[SEG];int n,tot;void pushup(int x){info[x]=info[ch[x][0]]*info[ch[x][1]];}void _build(int x,int l,int r){info[x].init(2);if (l==r){info[x].a[0][0]=g[lis[l]];info[x].a[0][1]=val[lis[l]];info[x].a[1][0]=info[x].a[1][1]=0;return;}int mid=l+r>>1;ch[x][0]=++tot; _build(ch[x][0],l,mid);ch[x][1]=++tot; _build(ch[x][1],mid+1,r);pushup(x);}void build(int _n){n=_n;tot=1;_build(1,1,n);}void _update(int x,int d,int lx,int rx){if (lx==rx){info[x].a[0][0]=g[lis[lx]];info[x].a[0][1]=val[lis[lx]];info[x].a[1][0]=info[x].a[1][1]=0;return;}int mid=lx+rx>>1;if (d<=mid) _update(ch[x][0],d,lx,mid);else _update(ch[x][1],d,mid+1,rx);pushup(x);}void update(int d){_update(1,d,1,n);}Mtrix _query(int x,int l,int r,int lx,int rx){if (l<=lx&&rx<=r) return info[x];int mid=lx+rx>>1;if (r<=mid) return _query(ch[x][0],l,r,lx,mid);else if (l>mid) return _query(ch[x][1],l,r,mid+1,rx);else{return _query(ch[x][0],l,mid,lx,mid)*_query(ch[x][1],mid+1,r,mid+1,rx);}}Mtrix query(int l,int r){return _query(1,l,r,1,n);}
}/*}}}*/
void add(int x,int y){a[++tot].y=y; a[tot].nxt=h[x]; h[x]=tot;}
void dfs(int fa,int x){int u;sz[x]=1; son[x]=0; pre[x]=fa;for (int i=h[x];i!=-1;i=a[i].nxt){u=a[i].y;if (u==fa) continue;dfs(x,u);sz[x]+=sz[u];if (sz[u]>sz[son[x]]) son[x]=u;}
}
void dfs1(int fa,int x){int u,cntson=0;dfn[x]=ed[x]=++dfn_t; lis[dfn_t]=x;if (son[x]){top[son[x]]=top[x];dfs1(x,son[x]);ed[x]=ed[son[x]];f[x]+=f[son[x]];++cntson;}for (int i=h[x];i!=-1;i=a[i].nxt){u=a[i].y;if (u==fa||u==son[x]) continue;top[u]=u;dfs1(x,u);g[x]+=f[u];++cntson;}if (cntson==0) g[x]=inf,f[x]=val[x];elsef[x]=min(val[x],f[x]+g[x]);
}
ll query(int x){Mtrix tmp=Seg::query(dfn[x],ed[x]);return min(tmp.a[0][0],tmp.a[0][1]);
}
void update(int x,int delta){ll tmp;val[x]+=delta;while (top[x]){tmp=query(top[x]);if (pre[top[x]])g[pre[top[x]]]-=tmp;Seg::update(dfn[x]);tmp=query(top[x]);if (pre[top[x]])g[pre[top[x]]]+=tmp;x=pre[top[x]];}
}int main(){
#ifndef ONLINE_JUDGEfreopen("a.in","r",stdin);
#endifchar op[5];int x,y;scanf("%d",&n);for (int i=1;i<=n;++i) scanf("%lld",val+i);memset(h,-1,sizeof(h));tot=0;for (int i=1;i<n;++i){scanf("%d%d",&x,&y);add(x,y); add(y,x);}dfs(0,1);top[1]=1; dfn_t=0;dfs1(0,1);Seg::build(dfn_t);scanf("%d",&m);for (int i=1;i<=m;++i){scanf("%s",&op);if (op[0]=='C'){scanf("%d%d",&x,&y);update(x,y);}else{scanf("%d",&x);ans=query(x);printf("%lld\n",ans);}}
}

转载于:https://www.cnblogs.com/yoyoball/p/9513666.html

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

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

相关文章

[USACO]地震 (二分答案+最优比率生成树详解)

题面&#xff1a;[USACO 2001 OPEN]地震 题目描述&#xff1a; 一场地震把约翰家的牧场摧毁了&#xff0c; 坚强的约翰决心重建家园。 约翰已经重建了N个牧场&#xff0c;现在他希望能修建一些道路把它们连接起来。研究地形之后&#xff0c;约翰发现可供修建的道路有M条。碰巧的…

HTTP协议学习笔记

1.HTTP协议简介 &#xff08;1&#xff09;客户端连上web服务器后&#xff0c;若想获得web服务器中的某个web资源&#xff0c;需遵守一定的通讯格式&#xff0c;HTTP协议用于定义客户端与web服务器通迅的格式。 &#xff08;2&#xff09;HTTP是hypertext transfer protocol&…

defer和async的原理与区别

上一篇刚转载了一篇有关于网站性能优化的文章&#xff0c;其中提及到了页面的加载和渲染的过程&#xff0c;提到了defer和async的相关区别&#xff0c;但是本人在此之前并没有深究其中的区别。 defer和async是script标签的两个属性&#xff0c;用于在不阻塞页面文档解析的前提…

一些奇妙的线段树操作

学过数据结构和会做题完全是两个概念orz 各种各样的题目都应该见识一下 简单的目录&#xff1a; 最大连续长度 吉司机线段树 线段树合并/分裂 最大连续长度问题 典型题目&#xff1a;HDU 3911 &#xff08;$Black$ $And$ $White$&#xff09; 题目大意&#xff1a;有一个长度为…

微服务实践沙龙-上海站

微服务的概念最早由Martin Fowler与James Lewis于2014年共同提出&#xff0c;核心思想是围绕业务能力组织服务&#xff0c;各个微服务可被独立部署&#xff0c;服务间是松耦合的关系&#xff0c;以及数据和治理的去中心化管理。微服务能够帮助企业应对业务复杂、频繁更新以及团…

Spring的refresh()方法调用过程

Spring的refresh()方法调用过程 refresh()是Spring中比较核心的方法&#xff0c;Spring所有的初始化都在这个方法中完成 具体代码如下 public void refresh() throws BeansException, IllegalStateException {synchronized (this.startupShutdownMonitor) {// Prepare this co…

Web数据存储之localStorage和sessionStorage

Web数据存储之localStorage和sessionStorage 学习前端以来&#xff0c;自己了解有localStorage和sessionStorage的相关存储的知识&#xff0c;也有实践过&#xff0c;但是之前只限于能用的基础上&#xff0c;但最近看了一本书&#xff0c;深入了解了localStorage和sessionStor…

(四)RabbitMQ消息队列-服务详细配置与日常监控管理

&#xff08;四&#xff09;RabbitMQ消息队列-服务详细配置与日常监控管理 原文:&#xff08;四&#xff09;RabbitMQ消息队列-服务详细配置与日常监控管理RabbitMQ服务管理 启动服务&#xff1a;rabbitmq-server -detached【 /usr/local/rabbitmq/sbin/rabbitmq-server -deta…

oracle中delete、truncate、drop的区别 (转载)

一、delete 1、delete是DML&#xff0c;执行delete操作时&#xff0c;每次从表中删除一行&#xff0c;并且同时将该行的的删除操作记录在redo和undo表空间中以便进行回滚&#xff08;rollback&#xff09;和重做操作&#xff0c;但要注意表空间要足够大&#xff0c;需要手动提交…

前端开发工程化探讨--基础篇(长文)

转载自UC资深前端工程师张云龙的github 喂喂喂&#xff0c;那个切图的&#xff0c;把页面写好就发给研发工程师套模板吧。 你好&#xff0c;切图仔。 不知道你的团队如何定义前端开发&#xff0c;据我所知&#xff0c;时至今日仍然有很多团队会把前端开发归类为产品或者设计岗…

Python读取Json字典写入Excel表格的方法

需求&#xff1a; 因需要将一json文件中大量的信息填入一固定格式的Excel表格&#xff0c;单纯的复制粘贴肯定也能完成&#xff0c;但是想偷懒一下&#xff0c;于是借助Python解决问题。 环境&#xff1a; Windows7 Python2.7 Xlwt 具体分析&#xff1a; 原始文件为json列表&am…

Spring-BeanFactory源码分析

正式进入Spring 源码分析这个模块了&#xff0c;对于spring这个庞大的工程&#xff0c;如果要一点点的完全分析是非常困难的&#xff0c;对于应用型框架&#xff0c;我还是偏向于掌握思想或者设计&#xff0c;而不是记住代码&#xff0c;对于初次看spring源码&#xff0c;相信大…

Linux查看修改时间、时区

同步网络时间 yum install ntpntpdate time.nist.gov timedatectl set-timezone Asia/Shanghai如果上面time.nist.gov服务器同步不了&#xff0c;可以换下面几个时间服务器试试&#xff1a;time.nist.govtime.nuri.net0.asia.pool.ntp.org1.asia.pool.ntp.org2.asia.pool.ntp.o…

我所知道的HTTP和HTTPS

摘要&#xff1a;相比之前的传输协议&#xff0c;HTTP/2在底层方面做了很多优化。有安全、省时、简化开发、更好的适应复杂页面、提供缓存利用率等优势&#xff0c;阿里云早在去年发布的CDN6.0服务就已正式支持HTTP/2&#xff0c;访问速度最高可提升68%。 写在前面 超文本传输…

sql server常用性能计数器

https://blog.csdn.net/kk185800961/article/details/52462913?utm_sourceblogxgwz5 https://blog.csdn.net/kk185800961/article/details/27657239 以下部分转自&#xff1a;http://www.cnblogs.com/zhijianliutang/p/4174697.html 常规计数器 收集操作系统服务器的服务器性能…

Python中正反斜杠('/'和'\')的意义

刚刚在学习些测试报告的时候&#xff0c;出现一个路径的问题&#xff0c;找了很久的原因&#xff0c;竟然是少了一个反斜杠引起的&#xff0c;在此顺便记录一下正反斜杠的作用。 在Python中&#xff0c;记录路径时有以下几种写法&#xff0c;如&#xff1a;&#xff08;大家都知…

什么是IOC容器

1.IOC不是一种技术&#xff0c;只是一种思想&#xff0c;一个重要的面向对象编程的法则&#xff0c;它能指导我们如何设计出松耦合&#xff0c;更优良的程序。传统应用程序都是由我们在类内部主动创建依赖对象&#xff0c;从而导致类与类之间高耦合&#xff0c;难于测试&#x…

Jenkins配置与使用

Jenkins是一个开源软件项目&#xff0c;旨在提供一个开放易用的软件平台&#xff0c;使软件的持续集成变成可能。Jenkins是基于Java开发的一种持续集成工具&#xff0c;用于监控持续重复的工作&#xff0c;功能包括&#xff1a;1、持续的软件版本发布/测试项目。2、监控外部调用…

fastDFS使用

fastDFS : 分布式文件系统C语言开发,fastDFS为互联网量身定制,考虑到了冗余备份,负载均衡,线性扩容...很容易搭建集群文件存储系统.存储在fastDFS图片:相当于存储在本地磁盘一样访问图片:相当于访问本地磁盘存储结构:组名/虚拟磁盘路径/动态生成文件名.扩展名192.168.100.20/gr…