专栏文章

题解:P8950 [YsOI2022] 道路修建

P8950题解参与者 2已保存评论 1

文章操作

快速查看文章及其快照的属性,并进行相关操作。

当前评论
1 条
当前快照
1 份
快照标识符
@miqtpt0r
此快照首次捕获于
2025/12/04 10:35
3 个月前
此快照最后确认于
2025/12/04 10:35
3 个月前
查看原文
好题,补一下题解。
注意到对于 k=1k=1 的情况不弱于最小树形图的板子。
回顾一下朱刘算法的过程:我们维护了一个内向树森林,每次取出一个没有出度的根节点 uu,找到它的最小出边 vv,则有两种情况:
  1. uuvv 不连通,直接连起来即可。
  2. u,vu,v 本来就联通,此时 vuv\to u 这一条链上的所有点就在一个环上,此时我们可以把这个环缩成点。
但是缩点之后这个环内多了一条边。考虑环内的点 uu,设它延伸出去的边长度为 dd,则如果别的边从 uu 连出去,则需要断开 dd 这条边。所以我们需要枚举 uu 的所有边,并且将其长度减去 dd,这样就正确了。
用带懒标记的可并堆去快速找到每个点的最小出边,复杂度为 O(nlogn)O(n\log n)
考虑拆贡献,计算每条边被包含的次数。
设我们现在考虑到 uvu\to v 这条边。记 SuS_u 表示 uu 节点包含的原图中的点的集合,则如果没有点在 SuS_u 内,则 SuS_u 中的点就会从这条边中走出去,贡献系数为 (nSuk)(nk)\dfrac {\binom{n-|S_u|}{k} }{\binom{n}{k}}
判断无解也是简单的:如果最终发现有一个根节点 uu 满足 Su+kn|S_u|+k\le n,则我们可以把这 kk 个节点全部扔到 SuS_u 的外面,这样就寄了。
CPP
#include<bits/stdc++.h>
using namespace std;

typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
namespace io{

template<typename T>
inline void gi(T &x)
{
	x=0;char ch=getchar();
	while(ch<'0'||'9'<ch) ch=getchar();
	while('0'<=ch&&ch<='9') x=x*10+ch-'0',ch=getchar();
}
template<typename T>
void print(T x)
{
	if(x<=9) return putchar(x+'0'),void();
	print(x/10),putchar(x%10+'0');
}

}using io::gi;using io::print;


const int N=5e5+9;
const int mod=998244353;

int fac[N],inv[N],ifac[N];
void ycl(int lim=2e5)
{
	fac[0]=fac[1]=inv[0]=inv[1]=ifac[0]=ifac[1]=1;
	for(int i=2;i<=lim;i++)
	{
		fac[i]=1ll*fac[i-1]*i%mod;
		inv[i]=1ll*(mod-mod/i)*inv[mod%i]%mod;
		ifac[i]=1ll*ifac[i-1]*inv[i]%mod;
	}
}
int C(int x,int y){return x>=y&&y>=0?1ll*fac[x]*ifac[y]%mod*ifac[x-y]%mod:0;}
int invC(int x,int y){return x>=y&&y>=0?1ll*ifac[x]*fac[y]%mod*fac[x-y]%mod:0;}

pii val[N];
int siz[N],son[N][2],tag[N],dep[N],tcnt=0;
int new_node(pii x)
{
	int u=++tcnt;
	siz[u]=1;tag[u]=son[u][0]=son[u][1]=0;
	val[u]=x;
	return u;
}
void Add(int u,int x){if(u) val[u].first+=x,tag[u]+=x;}
void pushdown(int u){if(tag[u]) Add(son[u][0],tag[u]),Add(son[u][1],tag[u]),tag[u]=0;}
void pushup(int u){siz[u]=siz[son[u][0]]+siz[son[u][1]]+1,dep[u]=dep[son[u][1]]+1;}
int Merge(int u,int v)
{
	if(!u||!v) return u+v;
	pushdown(u),pushdown(v);
	if(val[u]>val[v]) swap(u,v);
	son[u][1]=Merge(son[u][1],v);
	if(dep[son[u][0]]<dep[son[u][1]]) swap(son[u][0],son[u][1]);
	return pushup(u),u;
}
void pop(int &rt){pushdown(rt);rt=Merge(son[rt][0],son[rt][1]);}

int n,m,K;

int rt[N];

int nxt[N],len[N],tp[N];
int siz1[N];
int gettp(int x){return tp[x]==x?x:tp[x]=gettp(tp[x]);}

int bcj[N];
int getfa(int x){return bcj[x]==x?x:bcj[x]=getfa(bcj[x]);}

signed main()
{
	ycl();
	gi(n),gi(m),gi(K);
	for(int i=1;i<=m;i++)
	{
		int u,v,w;
		gi(u),gi(v),gi(w);
		if(u!=v)
		rt[u]=Merge(rt[u],new_node(make_pair(w,v)));
	}
	for(int i=1;i<=n;i++) bcj[i]=tp[i]=i;
	for(int i=1;i<=n;i++) siz1[i]=1;
	int ans=0;
	for(int u=1;u<=n;u++)
	{
		while(siz[rt[u]])
		{
			int v=val[rt[u]].second;
			if(gettp(u)==gettp(v)) {pop(rt[u]);continue;}
			(ans+=1ll*C(n-siz1[gettp(u)],K)*(val[rt[u]].first%mod+mod)%mod)%=mod;
			if(getfa(u)==getfa(v))
			{
				int uu=gettp(v),d=val[rt[u]].first;
				pop(rt[u]);
				Add(rt[u],-d);
				while(uu!=u)
				{
					Add(rt[uu],-len[uu]);
					rt[u]=Merge(rt[u],rt[uu]);
					uu=gettp(nxt[uu]);
				}
				uu=gettp(v);
				while(uu!=u)
				{
					siz1[u]+=siz1[uu];
					bcj[uu]=tp[uu]=u;
					uu=gettp(nxt[uu]);
				}
			}
			else
			{
				nxt[u]=gettp(v),len[u]=val[rt[u]].first;
				bcj[u]=v;
				break;
			}
		}
		if(!siz[rt[u]])
		{
			if(siz1[u]+K<=n) return puts("-1"),0;
			continue;
		}
	}
	print(1ll*ans*invC(n,K)%mod);
}

评论

1 条评论,欢迎与作者交流。

正在加载评论...