首页 > 其他分享 >[ZJOI2019] 语言 题解

[ZJOI2019] 语言 题解

时间:2024-04-21 17:22:19浏览次数:25  
标签:语言 int 题解 void tp fa sn ZJOI2019 id

不愧是 \(ZJOI\),《最可做的一道题》都让人一头雾水……


首先将问题转化到链上。

可以将总共的组数转化为每个点可以到达的城市。

明显给每个点建一棵动态开点线段树,维护可以和他通商的点。很明显,可以通商的点的标号连续的一段。我们可以将可以将每一次传播语言的工作当作区间修改,很明显可以用差分。最后再用线段树合并从后往前计算出每一个点的答案。

那假如问题转化到树上呢?

众所周知,假如我们想要让一棵树变成多个 木棍 链,树链剖分就是我们要熟悉掌握的一个知识点。

用树链剖分的方法,就可以最多进行 \(\log_2n\) 次操作,将所有本次被传播语言的点加入线段树。

时间复杂度 \(O(n\log^2n)\)。

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int N=1e5+5;
const int M=2e7+5;
int n,m,k,id,d;ll ans;
int h[N],to[N*2],nxt[N*2];
int ls[M],rs[M],rt[N],v[M],lb[M];
int dep[N],fa[N],sn[N];
int dfn[N],sz[N],tp[N];
struct line{int x,y;};
struct que{int x,y,z;};
vector<line>a;
vector<que>q[N];
void ad(int x,int y){
	to[++k]=y;
	nxt[k]=h[x];
	h[x]=k;
}void push_up(int x,int l,int r){
	if(lb[x]) v[x]=r-l+1;
	else v[x]=v[ls[x]]+v[rs[x]];
}void add(int &p,int l,int r,int x,int y,int kk){
	if(!p) p=++d;
	if(x<=l&&r<=y) lb[p]+=kk;
	else{
		int mid=(l+r)/2;
		if(x<=mid) add(ls[p],l,mid,x,y,kk);
		if(y>mid) add(rs[p],mid+1,r,x,y,kk);
	}push_up(p,l,r);
}int merge(int x,int y,int l,int r){
	if(!x||!y) return x|y;
	lb[x]+=lb[y];
	if(l<r){
		int mid=(l+r)/2;
		ls[x]=merge(ls[x],ls[y],l,mid);
		rs[x]=merge(rs[x],rs[y],mid+1,r);
	}push_up(x,l,r);
	return x;
}void dfs1(int x,int f){
	int mx=0;
	dep[x]=dep[f]+1;
	sz[x]=1;fa[x]=f;
	for(int i=h[x];i;i=nxt[i]){
		int y=to[i];
		if(y==f) continue;
		dfs1(y,x);
		if(sz[y]>mx){
			sn[x]=y;
			mx=sz[y];
		}sz[x]+=sz[y];
	}
}void dfs2(int x,int f){
	dfn[x]=++id;tp[x]=f;
	q[x].push_back({id,id,1});
	if(fa[x]) q[fa[x]].push_back({id,id,-1});
	if(!sn[x]) return;
	dfs2(sn[x],f);
	for(int i=h[x];i;i=nxt[i])
		if(to[i]!=fa[x]&&to[i]!=sn[x])
			dfs2(to[i],to[i]);
}int lca(int x,int y){
	a.clear();
	while(tp[x]!=tp[y]){
		if(dep[tp[x]]<dep[tp[y]]) swap(x,y);
		a.push_back({dfn[tp[x]],dfn[x]});
		x=fa[tp[x]];
	}if(dep[x]>dep[y]) swap(x,y);
	a.push_back({dfn[x],dfn[y]});
	return x;
}void solve(int x){
	for(int i=h[x];i;i=nxt[i]){
		int y=to[i];
		if(y==fa[x]) continue;
		solve(y);
		rt[x]=merge(rt[x],rt[y],1,n);
	}for(int i=0;i<q[x].size();i++)
		add(rt[x],1,n,q[x][i].x,q[x][i].y,q[x][i].z);
	ans+=v[rt[x]]-1;
}int main(){
	ios::sync_with_stdio(0);
	cin.tie(0);cout.tie(0);
	cin>>n>>m;
	for(int i=1,x,y;i<n;i++){
		cin>>x>>y;
		ad(x,y);ad(y,x);
	}dfs1(1,0);dfs2(1,1);
	while(m--){
		int x,y,lc;
		cin>>x>>y;
		lc=fa[lca(x,y)];
		for(int i=0;i<a.size();i++){
			q[x].push_back({a[i].x,a[i].y,1});
			q[y].push_back({a[i].x,a[i].y,1});
			if(lc) q[lc].push_back({a[i].x,a[i].y,-2});
		}
	}solve(1);
	cout<<ans/2;
	return 0;
}

标签:语言,int,题解,void,tp,fa,sn,ZJOI2019,id
From: https://www.cnblogs.com/chang-an-22-lyh/p/18149210/zjoi2019_yuyan_tj

相关文章

  • 在C语言中如何找到字符串的长度
    在C语言中处理字符串时,你需要知道如何找到它们的长度。在许多情况下,找到C语言中字符串的长度都是至关重要的。你可能需要执行字符串操作,而许多字符串操作函数都需要字符串的长度作为参数。你可能还需要验证用户输入、比较两个字符串,或者动态管理和分配内存。在本文中,你将学习在......
  • c语言程序设计——实验报告六
    实验项目名称:实验6循环结构程序设计(for语句的应用)实验项目类型:验证性实验日期:2024年4月15日一、实验目的1.熟练掌握三种循环语句并能正确运用;2.能够用循环实现一些常用算法,如穷举法,迭代法,递推法等;3.进一步学习程序调试;4.了解中国算法,百钱买百鸡。二、实验硬、软件环境W......
  • c语言程序设计——实验报告五
    实验项目名称:实验5循环结构程序设计(while、do-while语句的应用)实验项目类型:验证性实验日期:2024年4月11日一、实验目的1.熟练掌握三种循环语句并能正确运用;2.能够用循环实现一些常用算法,如穷举法,迭代法,递推法等;3.进一步学习程序调试;4.了解中国算法,百钱买百鸡。二、实验......
  • c语言操作符详解
    1,算数操作符%两个操作数必须为整数/2,移位操作符(只作用于整数)整数的二进制表示有原码,反码,补码,存储到内存的是补码,用的也是补码进行运算举例10000000000000000000000000000001------原码11111111111111111111111111111110    ------反码正数的反码是其本身,负数的......
  • 初识C语言
    一,c语言的基本概述  c语言是一门编译型语言。c语言的程序需要通过编译软件程序将源代码转换成可执行代码的程序,可执行代码是用计算机机器语言表示的代码。如:c语言在Windows下生成的可执行代码是以.exe为后缀的,在Unix下生成的可执行代码是以.out为后缀的。二,c语言的特性  c......
  • 【题解】[NOIP2001 普及组] 装箱问题
    [NOIP2001普及组]装箱问题这是一道动态规划题。那就先定义状态吧(这里用的是一维滚动数组)。$f[j]$代表当我有$j$这么多容量可以用时,能装的最大重量是多少。好,状态定义好了再想状态转移方程。$f[j]$可以从哪里转移过来呢?想一想,当我们循环到第$i$个物品时,我们面临两个......
  • 82.8K Star 功能强大的语言处理的PYTHON库
    简介LangChain是一个框架,用于开发由大型语言模型(LLMs)提供支持的应用程序。langchain库是功能强大的语言处理工具,可以用于文本处理、语言分析等多种任务。本文将介绍该库的安装、特性、基本功能、高级功能、实际应用场景,并进行总结。特性多语言支持:支持多种语言的处理和分......
  • 2024-04-21:用go语言,给一棵根为1的树,每次询问子树颜色种类数。 假设节点总数为n,颜色总
    2024-04-21:用go语言,给一棵根为1的树,每次询问子树颜色种类数。假设节点总数为n,颜色总数为m,每个节点的颜色,依次给出,整棵树以1节点做头,有k次查询,询问某个节点为头的子树,一共有多少种颜色。1<=n,m,k<=10^5。答案2024-04-21:来自左程云。chatgpt大体步骤如下:大体过程描述......
  • 简单的数学题 题解
    题意:解方程\[a^x\equivx\pmodm\]数据范围:\(a<m\le10^9\)Solution首先\(a^x\equiva^{x\bmod\varphi(m)+\varphi(m)}\pmodm\)我们设\(\text{solve}(\&x,y,m)\)表示解决\[a^{x\bmod\varphi(m)+y}\equivx\pmodm\]原题即\(\text{solve}(\&x,......
  • 数表 题解
    “当你想不出来一道题的时候就想一下排序”先不考虑\(a\),求\[\begin{aligned}&\sum_{i=1}^n\sum_{j=1}^m\sigma_1(\gcd(i,j))\\=&\sum_{i=1}^n\sum_{j=1}^m\sum_{d=1}^{\min(n,m)}[d=\gcd(i,j)]\sigma_1(d)\\=&\sum_{d=1}^{\min(n,m)}\sigma_1(d)\sum_......