\[ \texttt{Preface} \]
这题在 \(\text{Luogu}\) 上竟然不能交 \(C++\) ,会一直 \(Waiting\) ,只能交非 \(C++\) 的语言。
所以打完了 \(C++\) 要转到 \(C\) 才能过。
要把什么 \(swap\) , \(max\) 各种函数换成手写,以及 \(C++\) 的特色(例如 using namespace std;
和 inline
)都要去掉。
详情见 \(Code\) 。
\[ \texttt{Description} \]
给出一个 \(n\) 个点的带权树,需要支持以下操作:
CHANGE i ti
将第 \(i\) 条边的权值改为 \(t_i\) 。QUERY a b
询问 \(a\) 到 \(b\) 的路径上最大边权。
多组数据。
\[ \texttt{Solution} \]
从这个询问 " 查询路径信息,边带修 " 来说,我们可以知道这是一个树剖板子题。
不了解树剖的童鞋可以去了解一下,过一下 树剖模板 。
只不过这题不是一般的 " 点带修 " 而是 " 边带修 " ,也不要紧。
注意到除了根,每个节点都有父亲,那么我们可以把边的信息转化到点身上,每个节点的点权是它与它父亲所形成的边的边权。
例如 \(1\) 到 \(2\) 的一条长度为 \(3\) 的边(此时 \(1\) 是 \(2\) 的父亲),那么我们可以理解为 \(2\) 的点权是 \(3\) 。
这样就可以用树剖维护了。
但是令 \(z=\text{lca}(x,y)\) ,我们发现 \((fa[z],z)\) 这条边是不能被算进答案的。
在查询的最后一步,\(x\) 和 \(y\) 会在同一条重链上(设 \(dep_x
\[ \texttt{Code} \]
#include
#define N 10100
#define M 20100
int read()
{
int x=0,f=1;char s=getchar();
while(s<'0'||s>'9'){if(s=='-')f=-f;s=getchar();}
while(s>='0'&&s<='9'){x=x*10+s-'0';s=getchar();}
return x*f;
}
int tmp;
int max(int a,int b){return a>b?a:b;}
int T;
int n;
struct Edge{
int u,v,w;
}e[N];
int tot,head[N],ver[M],edge[M],Next[M];
void add(int u,int v,int w)
{
ver[++tot]=v; edge[tot]=w; Next[tot]=head[u]; head[u]=tot;
}
int val[N];
int d[N];
int fu[N];
int size[N];
int son[N];
void dfs1(int u)
{
size[u]=1;
for(int i=head[u];i;i=Next[i])
{
int v=ver[i],w=edge[i];
if(v==fu[u])continue;
fu[v]=u;
val[v]=w;
d[v]=d[u]+1;
dfs1(v);
size[u]+=size[v];
if(size[son[u]]d[top[v]])tmp=u,u=v,v=tmp;
ans=max(ans,ask(1,dfn[top[v]],dfn[v]));
v=fu[top[v]];
}
if(u==v)return ans;
if(d[u]>d[v])tmp=u,u=v,v=tmp;
ans=max(ans,ask(1,dfn[u]+1,dfn[v]));
return ans;
}
void work()
{
tot=QwQ=0;
for(int i=1;i<=n;i++)
head[i]=son[i]=0;
n=read();
for(int i=1;id[e[x].v])
tmp=e[x].u,e[x].u=e[x].v,e[x].v=tmp;
change(1,dfn[e[x].v],y);
break;
}
case 'Q':{
printf("%d\n",path_ask(x,y));
break;
}
}
}
}
int main()
{
T=read();
while(T--) work();
return 0;
}
\[ \texttt{Thanks} \ \texttt{for} \ \texttt{watching} \]