博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
POJ 2763 (树链剖分+边修改+边查询)
阅读量:7158 次
发布时间:2019-06-29

本文共 2782 字,大约阅读时间需要 9 分钟。

题目链接:

题目大意:某人初始在s点。有q次移动,每次移动沿着树上一条链,每经过一条边有一定花费,这个花费可以任意修改。问每次移动的花费。

解题思路

树链剖分基础题。每次Q之后改变一下s。

线段树记录的是边权。方法是对于一条边(u,v),边权值加在dep比较大的那一端。

链查询(边)和 链查询(点)在轻链时略有不同。

注意本题使用vector邻接表存图是会TLE的,应该使用链式前向星。树链剖分中使用链式前向星是基本要求。

 

#include "cstdio"#include "cstring"using namespace std;#define maxn 100005 int s[maxn],dep[maxn],w[maxn],fa[maxn],top[maxn],son[maxn],val[maxn],cnt,tol,n;int head[maxn];struct Edge{    int u,v,c;}edge[maxn];struct EDGE{    int next,to;}e[2*maxn];void addedge(int u,int v){    e[tol].to=v;    e[tol].next=head[u];    head[u]=tol++;}void dfs1(int u,int pre,int d){    s[u]=1;fa[u]=pre;dep[u]=d;    for(int i=head[u];i!=-1;i=e[i].next)    {        int v=e[i].to;        if(v==pre) continue;        dfs1(v,u,d+1);        s[u]+=s[v];        if(son[u]!=-1||s[v]>s[son[u]]) son[u]=v;    }}void dfs2(int u,int tp){    w[u]=++cnt;top[u]=tp;    if(son[u]==-1) return;    dfs2(son[u],tp);    for(int i=head[u];i!=-1;i=e[i].next)    {        int v=e[i].to;        if(v!=son[u]&&v!=fa[u]) dfs2(v,v);    }}//Segment Tree#define lson l,mid,root<<1#define rson mid+1,r,root<<1|1int sum[maxn<<2];void PushUp(int root){    sum[root]=sum[root<<1]+sum[root<<1|1];}void Build(int l,int r,int root){    if(l==r)    {        sum[root]=val[l];        return;    }    int mid=(l+r)>>1;    Build(lson);    Build(rson);    PushUp(root);}void Update(int p,int v,int l,int r,int root){    if(l==r)    {        sum[root]=v;        return;    }    int mid=(l+r)>>1;    if(p<=mid) Update(p,v,lson);    else Update(p,v,rson);    PushUp(root);}int Query(int L,int R,int l,int r,int root){    if(L<=l&&r<=R) return sum[root];    int mid=(l+r)>>1;    int ret=0;    if(L<=mid) ret+=Query(L,R,lson);    if(R>mid) ret+=Query(L,R,rson);    return ret;}void Change(int x,int v){    if(dep[edge[x].u]>dep[edge[x].v]) Update(w[edge[x].u],v,1,n,1);    else Update(w[edge[x].v],v,1,n,1);}int query(int x,int y){    int ans=0;    while(top[x]!=top[y])    {        if(dep[top[x]]
dep[y]) swap(x,y); if(x!=y) ans+=Query(w[son[x]],w[y],1,n,1);// return ans;}int main(){ //freopen("in.txt","r",stdin); int m,s,u,t,cmd; while(~scanf("%d%d%d",&n,&m,&s)) { memset(son,-1,sizeof(son)); memset(head,-1,sizeof(head)); cnt=tol=0; for(int i=1;i
dep[edge[i].v]) val[w[edge[i].u]]=edge[i].c; else val[w[edge[i].v]]=edge[i].c; } Build(1,n,1); while(m--) { scanf("%d",&cmd); if(cmd==0) { scanf("%d",&t); printf("%d\n",query(s,t)); s=t; } if(cmd==1) { scanf("%d%d",&u,&t); Change(u,t); } } }}

 

13493821 Accepted 9676K 1579MS 3176B 2014-10-01 16:16:22

转载地址:http://kpegl.baihongyu.com/

你可能感兴趣的文章
哦屋~如此完美的富文本编辑器你值得拥有
查看>>
LeetCode 之 JavaScript 解答第226题 —— 翻转二叉树(Invert Binary Tree)
查看>>
去中心化应用的五大制胜关键
查看>>
ES6新特性
查看>>
DeepMind AI与人类合作玩夺旗策略游戏,表现与人类玩家相当
查看>>
iOS 使用wkwebview加载本地html出现ajax错误
查看>>
js阿拉伯数字转成汉字
查看>>
webpack配置proxyTable时pathRewrite无效的解决方法
查看>>
智能指针(理解以及实现)
查看>>
数据分析软件Power BI探索数据教程(一)——关于“快速见解”功能
查看>>
《云周刊》69期:开门红利!阿里云2月活动来袭
查看>>
从零开始搭建webpack+react开发环境
查看>>
js __proto__和prototype的关系
查看>>
[翻译]了解NodeJS看这一篇就够了
查看>>
Swift Package Manager使用总结
查看>>
iOS模拟器无法启动 unable to boot the simulator的几种解决方法
查看>>
纯 CSS 实现多行文字截断
查看>>
据说只有前端程序员才能看懂!
查看>>
(持续更新, 目前含100+工具类) DevUtils 是一个 Android 工具库
查看>>
JavaScript 复习之 Element 节点
查看>>