首页 > 其他分享 >P9168 [省选联考 2023] 人员调度

P9168 [省选联考 2023] 人员调度

时间:2023-11-22 21:23:41浏览次数:33  
标签:P9168 省选 siz tot 联考 int maxn mnp id

去年省选的时候还不会霍尔定理,想到了线段树分治想不了贪心。今年看感觉挺傻逼的。

先线段树分治,把删除操作扔了。如果我们要知道一个人最后扔到哪里,那就是一个费用流问题,不太可能解决,考虑用霍尔定理刻画这个东西,我们发现,最后一个人的集合能匹配上当且仅当:
计 \(u\) 子树里有 \(p_u\) 个人,那么满足 \(\forall u,siz_u-p_u \ge 0\)。

每次如果加入一个人,需要满足这个人到根的路径上所有 \(siz_u-p_u>0\)。如果满足可以直接加,否则,我们必须换掉一个人,设 \(u\) 到根路径上最深的 \(siz_u-p_u=0\) 的点为 \(w\),则换掉的点必须在 \(w\) 子树内。拿两颗线段树做这事就行。

复杂度 \(n\log^3{n}\)。这里默认 \(n,q\) 同阶。

#include <bits/stdc++.h>
using namespace std;
const int maxn=2e5+10;
typedef long long ll;
int n,m,k,sid,fat[maxn],tot,lt[maxn],rt[maxn];
ll x[maxn<<1],v[maxn<<1],ans;
int p[maxn];
vector<int> mdi[maxn<<2];
vector<int> G[maxn];
ll mn[maxn<<2],siz[maxn],son[maxn],top[maxn],dfn[maxn],times,seq[maxn],tag[maxn<<2];
set< pair<ll,int> >qu[maxn<<2];
ll mnt[maxn<<2],idt[maxn<<2];
inline void Pushup(int i)
{
	if(mnt[i<<1]<mnt[i<<1|1])
		mnt[i]=mnt[i<<1],idt[i]=idt[i<<1];
	else 
		mnt[i]=mnt[i<<1|1],idt[i]=idt[i<<1|1];
}
inline void addp(int i,int l,int r,int p,int v,int id)
{
	if(l==r)
	{
		qu[i].insert({v,id});
		if(qu[i].empty())mnt[i]=1e9,idt[i]=0;
		else 
			mnt[i]=(*qu[i].begin()).first,idt[i]=(*qu[i].begin()).second;
		return;
	}
	int mid=l+r>>1;
	if(p<=mid)addp(i<<1,l,mid,p,v,id);
	if(p>mid)addp(i<<1|1,mid+1,r,p,v,id);
	Pushup(i);
}
inline void clrp(int i,int l,int r,int p,int v,int id)
{
	if(l==r)
	{
		assert(qu[i].find({v,id})!=qu[i].end());
		qu[i].erase({v,id});
		if(qu[i].empty())mnt[i]=1e9,idt[i]=0;
		else 
			mnt[i]=(*qu[i].begin()).first,idt[i]=(*qu[i].begin()).second;
		return;
	}
	int mid=l+r>>1;
	if(p<=mid)clrp(i<<1,l,mid,p,v,id);
	if(p>mid)clrp(i<<1|1,mid+1,r,p,v,id);
	Pushup(i);
}
pair<int,int> tquery(int i,int l,int r,int L,int R)
{
	if(L<=l&&r<=R)return {mnt[i],idt[i]};
	int mid=l+r>>1;
	if(R<=mid)return tquery(i<<1,l,mid,L,R);
	if(L>mid)return tquery(i<<1|1,mid+1,r,L,R);
	return min(tquery(i<<1,l,mid,L,R),tquery(i<<1|1,mid+1,r,L,R));
}
inline void pushtg(int i,int v)
{
	mn[i]+=v;
	tag[i]+=v;
}
inline void pushdown(int i)
{
	if(!tag[i])return;
	pushtg(i<<1,tag[i]);
	pushtg(i<<1|1,tag[i]);
	tag[i]=0;
}
inline void update(int i,int l,int r,int L,int R,int v)
{
	if(L<=l&&r<=R){pushtg(i,v);return;}
	int mid=l+r>>1;pushdown(i);
	if(L<=mid)update(i<<1,l,mid,L,R,v);
	if(R>mid)update(i<<1|1,mid+1,r,L,R,v);
	mn[i]=min(mn[i<<1],mn[i<<1|1]);
}
inline int query(int i,int l,int r,int L,int R)
{
	if(L<=l&&r<=R)return mn[i];
	int mid=l+r>>1;pushdown(i);
	if(R<=mid)return query(i<<1,l,mid,L,R);
	if(L>mid)return query(i<<1|1,mid+1,r,L,R);
	return min(query(i<<1,l,mid,L,R),query(i<<1|1,mid+1,r,L,R));
}
inline int findz(int i,int l,int r,int L,int R)
{
	if(r<L||l>R)return 0;
	if(mn[i]>0)return 0;
	if(l==r)return l;
	
	int mid=l+r>>1;pushdown(i);
	int ret=0;
	if(R>mid)ret=findz(i<<1|1,mid+1,r,L,R);
	if(ret)return ret;
	return findz(i<<1,l,mid,L,R);
}
inline void updrt(int x,int v)
{
	while(x)
	{
		update(1,1,n,dfn[top[x]],dfn[x],v);
		x=fat[top[x]];
	}
}
inline int qryrt(int x)
{
	int ret=1e9;
	while(x)
	{
		int now=query(1,1,n,dfn[top[x]],dfn[x]);
		ret=min(ret,now);
		if(ret==0)return seq[findz(1,1,n,dfn[top[x]],dfn[x])];
		x=fat[top[x]];
	}
	return 0;
}
inline void build(int i,int l,int r)
{
	mnt[i]=1e9;
	if(l==r)
	{
		mn[i]=siz[seq[l]];
		return;
	}
	int mid=l+r>>1;
	build(i<<1,l,mid);
	build(i<<1|1,mid+1,r);
	mn[i]=min(mn[i<<1],mn[i<<1|1]);
}
inline void Modify(int i,int l,int r,int L,int R,int id)
{
	if(L<=l&&r<=R){mdi[i].push_back(id);return;}
	int mid=l+r>>1;
	if(L<=mid)Modify(i<<1,l,mid,L,R,id);
	if(R>mid)Modify(i<<1|1,mid+1,r,L,R,id);
}
void topdfs(int u,int topf)
{
	top[u]=topf;
	seq[dfn[u]=++times]=u;
	if(!son[u])return;
	topdfs(son[u],topf);
	for(int v:G[u])
		if(v!=son[u])
			topdfs(v,v);
}
ll Ans[maxn];
inline void dfs(int i,int l,int r)
{
	vector<int> opr;
	for(int id:mdi[i])
	{
		int mnp=qryrt(x[id]);
		if(!mnp) 
		{
			ans+=v[id];
			opr.push_back(id);
			updrt(x[id],-1);
			addp(1,1,n,dfn[x[id]],v[id],id);
		}
		else 
		{
			ll tmn,td;
			tie(tmn,td)=tquery(1,1,n,dfn[mnp],dfn[mnp]+siz[mnp]-1);
			if(tmn<v[id])
			{
				int pos=x[td];
				ans-=tmn;
				ans+=v[id];
				updrt(pos,1);
				clrp(1,1,n,dfn[pos],v[td],td);
				opr.push_back(id);
				opr.push_back(-td);
				updrt(x[id],-1);
				addp(1,1,n,dfn[x[id]],v[id],id);
			}
		}
	}
	if(l==r)
		Ans[l]=ans;
	else 
	{
		int mid=l+r>>1;
		dfs(i<<1,l,mid);
		dfs(i<<1|1,mid+1,r);
	}
	reverse(opr.begin(),opr.end());
	for(int id:opr)
	{
		if(id>0)ans-=v[id],updrt(x[id],1),clrp(1,1,n,dfn[x[id]],v[id],id);
		else ans+=v[-id],updrt(x[-id],-1),addp(1,1,n,dfn[x[-id]],v[-id],-id);
	}
}
int main()
{
	cin>>sid;
	cin>>n>>k>>m;
	for(int i=2;i<=n;i++)
		cin>>fat[i],G[fat[i]].push_back(i);
	for(int i=1;i<=n;i++)siz[i]=1;
	tot=k;
	for(int i=1;i<=k;i++)
		cin>>x[i]>>v[i],rt[i]=m;
	for(int i=1;i<=m;i++)
	{
		int opt;
		cin>>opt;
		if(opt==1)
		{
			++tot;
			cin>>x[tot]>>v[tot];
			lt[tot]=i;
			rt[tot]=m;
		}
		else 
		{
			int id;
			cin>>id;
			rt[id]=i-1;
		}
	}
	for(int i=n;i>=2;i--)
	{
		siz[fat[i]]+=siz[i];
		if(siz[son[fat[i]]]<siz[i])son[fat[i]]=i;
	}
	topdfs(1,1);
	for(int i=1;i<=tot;i++)
		Modify(1,0,m,lt[i],rt[i],i);
	build(1,1,n);
	dfs(1,0,m);
	for(int i=0;i<=m;i++)
		cout<<Ans[i]<<' ';
	cout<<'\n';
	return 0;
}

标签:P9168,省选,siz,tot,联考,int,maxn,mnp,id
From: https://www.cnblogs.com/hikkio/p/P9168.html

相关文章

  • 题解 「2019五校联考-镇海1」一棵树
    题意一棵\(n\)个结点的树,根节点为\(1\),结点\(i\)的父亲是\(f_i\)。\(f_1=f_0=0\)。对于每一个整数\(i\),假如\(f_{f_i}\)不为\(0\),那么就将\(f_{f_i}\)与\(i\)连上一条边。从每一个结点,每次随机向相邻的结点走。问每个结点期望走多少步才能走到根。对于\(60\%\),\(......
  • [省选联考 2021 A/B 卷] 滚榜 - 总结
    [省选联考2021A/B卷]滚榜首先明确要拆开\(\sumb_i\)。因为\(b_i\)不降,所以当一个人做了\(x\)道,后边的人都至少做了\(x\)道。然后我们考虑提前算贡献,对于一个排名\(p_i\)(倒序且\(p_0\)为\(a_i\)最大的位置),贡献是这样的:\(\sumb_i=\sum\max(0,\a_{p_{i-1}}-a_......
  • P7514 [省选联考 2021 A/B 卷] 卡牌游戏
    [省选联考2021A/B卷]卡牌游戏题目描述Alice有\(n\)张卡牌,第\(i\)(\(1\lei\len\))张卡牌的正面有数字\(a_i\),背面有数字\(b_i\),初始时所有卡牌正面朝上。现在Alice可以将不超过\(m\)张卡牌翻面,即由正面朝上改为背面朝上。Alice的目标是让最终朝上的\(n\)个数......
  • 【多校联考NOIP#12】比赛复盘
    A.星穹铁道读完题面就想到了\(O(n^2)\)的暴力。很好想,但是只有40分。观察到\(z_i=\pm1\),然而即便如此,我也没有得到有用的性质。(正解是用到这个性质的)然后我就暴力写了。正解的性质“最终在一个区间L,R内,初始也一定在一个连续段内”赛事没有想到。同时题解用了逆向思维,对......
  • 2023联合省选 题解
    目录D1T1P9166[省选联考2023]火车站D1T2P9167[省选联考2023]城市建造D1T3P9168[省选联考2023]人员调度D2T1P9169[省选联考2023]过河卒D2T2P9170[省选联考2023]填数游戏D2T3P9171[省选联考2023]染色数组D1T1P9166[省选联考2023]火车站性质很好找。关......
  • 一则求总贡献的例题——联考题引发的思考
    对于一些求总贡献的题,与许多人的常识相反,直接求期望往往比求总和更容易.以今天联考T1的一个环节为例.【例】对排列\(P_n\),定义\(C(P_n):=\left|\{i:P_j>P_i,\\forallj<i\}\right|\),即其前缀最小值序列的不同数个数.给定\(n\),求全部\(n!\)个排列\(P_n\)的\((-1)^{C(......
  • [十二省联考 2019] 春节十二响
    [十二省联考2019]春节十二响感觉作为例题还是挺不错的。感官上直接分析比较困难。不妨先考虑怎样的段长集合是合法的。注意到合法等价于对每一条从根到叶子的链都合法,考虑在链上贪心,尝试将每个和比他大的最小的点做匹配,如果能匹配上就是合法。很显然,如果仅考虑一条链,那么极小......
  • P3746 [六省联考 2017] 组合数问题
    看了题解才悟了,我还是太菜了。solution要求\[\left(\sum_{i=0}^\inftyC_{nk}^{ik+r}\right)\bmodp\]这个形式很像生成函数吧。我们套用生成函数:\[G(x)=\sum_{i=0}^{\infty}\begin{pmatrix}nk\\i\end{pmatrix}x^i\]所求即为\[\sum_{i\bmodk=r}[x^i](1+x)......
  • 「联合省选 2020 A」组合数问题 题解
    非常显然的,我们展开\(f(k)\),于是有:\[\begin{align}&\sum\limits_{k=0}^{n}\sum\limits_{i=0}^{m}a_{i}k^{i}x^{k}\binom{n}{k}\\=&\sum\limits_{k=0}^{n}\sum\limits_{i=0}^{m}a_{i}{\sum\limits_{j=0}^{i}\begin{Bmatrix}i\\j\end{Bmatrix}\binom{......
  • 省选联考 2022 填树
    洛谷传送门LOJ传送门这题做得真艰难。先考虑第一问。一眼看上去并没有什么复杂度脱离值域的办法。考虑枚举一个\(x\)表示最小值,那么点权只能在\([x,x+K]\)中。点权最小值不一定为\(x\),减去点权在\([x+1,x+K]\)中的答案即可,也就是把\(K\)减\(1\)后再算一......