题目大意:多次给出关键点,求切断边使所有关键点与1断开的最小费用
分析:每次造出虚树,dp[i]表示将i和i子树与父亲断开费用
对于父亲x,儿子y
①y为关键点:\(dp[x]\)+=\(dismn(x,y)\)
②y不为关键点:要么断y,要么断y所有子树
\(dp[x]\)+=\(min(dismn(x,y),dp[y])\)
=========================================================
关于兼容性的一种讨论
dismn(x,y)直接改为dismn(1,x)预处理算可以吗
当然不行?
交一发,A
兼容性:若要算到y,则y到1路径中没有关键点
①情况无影响
②情况中若子树中最小值算到了上面,断y定更优,而断y又变成父节点的①②情况讨论
所以是可以的咯
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <cctype>
using namespace std;
typedef long long LL;
const int M=250007;
inline int rd(){
int x=0;bool f=1;char c=getchar();
for(;!isdigit(c);c=getchar())if(c=='-')f=0;
for(;isdigit(c);c=getchar())x=x*10+c-48;
return f?x:-x;
}
int n,m;
LL dp[M];
LL dis[M];
LL w[M][20];
int ln[M];
int dep[M],sz[M],son[M],pre[M];
int top[M],dfn[M],pid[M],T;
int que[M],kd[M];
bool cmp(int x,int y){return dfn[x]<dfn[y];}
int st[M],tot;
int g[M],te;
struct edge{int y,next;LL d;}e[M<<1];
void addedge(int x,int y,LL d){
e[++te].y=y;e[te].d=d;e[te].next=g[x];g[x]=te;
}
int hd[M],td;
struct link{int y,next;}dw[M];
void addlink(int x,int y){
if(x==y)return;
dw[++td].y=y;dw[td].next=hd[x];hd[x]=td;
}
void dfs1(int x){
sz[x]=1;
int p,y;
for(p=g[x];p;p=e[p].next)
if((y=e[p].y)!=pre[x]){
dep[y]=dep[x]+1;
dis[y]=e[p].d;
pre[y]=x;
dfs1(y);
sz[x]+=sz[y];
if(sz[y]>sz[son[x]]) son[x]=y;
}
}
void dfs2(int x){
pid[dfn[x]=++T]=x;
if(son[x]){
top[son[x]]=top[x];
dfs2(son[x]);
}
int p,y;
for(p=g[x];p;p=e[p].next)
if((y=e[p].y)!=pre[x]&&y!=son[x]){
top[y]=y;
dfs2(y);
}
}
int LCA(int x,int y){
while(top[x]!=top[y]){
if(dep[top[x]]<dep[top[y]]) swap(x,y);
x=pre[top[x]];
}
if(dep[x]>dep[y]) swap(x,y);
return x;
}
void vbuild(int z){
int i,x,anc;
sort(que+1,que+z+1,cmp);
for(i=1;i<z;i++){
anc=LCA(que[i],que[i+1]);
hd[anc]=0; kd[anc]=2; dp[anc]=0;
}
hd[1]=0; dp[1]=0;//*****
for(i=1;i<=z;i++){
x=que[i];
hd[x]=0; kd[x]=1; dp[x]=0;
}
td=0;
tot=0;
st[++tot]=1;
for(i=1;i<=z;i++){
x=que[i];
anc=LCA(x,st[tot]);
if(anc==st[tot]){
st[++tot]=x;
continue;
}
else{
while( tot>1 && dep[anc]<=dep[st[tot-1]]){
addlink(st[tot-1],st[tot]);
tot--;
}
addlink(anc,st[tot]);
st[tot]=anc;
st[++tot]=x;
}
}
for(i=1;i<tot;i++) addlink(st[i],st[i+1]);
}
LL getm(int x,int y){
int l=ln[y-x+1];
return min(w[x][l],w[y-(1<<l)+1][l]);
}
void init(){
int i,j,l;
for(i=2;i<=n;i++) ln[i]=ln[i>>1]+1;
for(i=1;i<=n;i++) w[i][0]=dis[pid[i]];
for(i=n;i>0;i--){
l=ln[n-i+1];
for(j=1;j<=l;j++) w[i][j]=min(w[i][j-1],w[i+(1<<j-1)][j-1]);
}
}
LL getw(int x,int y){
LL res=1LL<<61;
while(dep[top[x]]>dep[y]){
res=min(res,getm(dfn[top[x]],dfn[x]));
x=pre[top[x]];
}
if(x!=y) res=min(res,getm(dfn[y]+1,dfn[x]));
return res;
}
void dfs(int x){
int p,y;
LL tp;
for(p=hd[x];p;p=dw[p].next){
y=dw[p].y;
tp=getw(y,x);
if(kd[y]==1) dp[x]+=tp;
else{
dfs(y);
dp[x]+=min(tp,dp[y]);
}
}
}
int main(){
int i,x,y,z;
n=rd();
for(i=1;i<n;i++){
x=rd(),y=rd(),z=rd();
addedge(x,y,z);
addedge(y,x,z);
}
dep[1]=pre[1]=0;
dis[1]=0;
dfs1(1);
top[1]=1;
dfs2(1);
init();
m=rd();
while(m--){
z=rd();
for(i=1;i<=z;i++) que[i]=rd();
vbuild(z);
dfs(1);
printf("%lld\n",dp[1]);
}
return 0;
}