【PKUSC2019】树染色【线段树合并】【树形DP】
Description
给出一棵n个点的树,现在有m种颜色,要给每个节点染色,相邻节点不能同色。
另外有k条限制,形如x号点不能为颜色y
同一节点有可能有多条限制。
求方案数对998244353取模的结果。
n<=200000,m<=1e9,k<=400000
Solution
考场上一直在想怎么容斥做,怎么都弄不出来。
学傻了。
考虑暴力DP
设\(f[i][j]\)为当前处理了以i为根的子树,i的颜色为j的方案数。
记\(g[i]=\sum\limits_{k}f[i][k]\)
显然有转移$$f[i][j]=[!ban[i][j]]\prod_{p\in son[i]}(g[p]-f[p][j])$$
但是这样的状态数是\(n*m\)的,我们发现只需要记下子树中有的颜色,其他的颜色的答案都是一样的。
这样状态数缩减到\(n*k\),但还是很大,于是我们考虑采用线段树来维护。
转移的时候我们将子树一个个的合并到根
大概是\(f[i][j]=(g[p]-f[p][j])*f[i][j]\)
根据这个我们就可以线段树合并了。
如果只有父亲有,就直接乘
儿子父亲都有暴力合并
只有儿子有的话把括号拆开,就是乘上\(-f[i][j]\)加上\(g[p]*f[i][j]\)
需要维护区间乘区间加,类似一次函数维护即可。
时间复杂度大概是\(O((n+k)\log m)\),具体可以看代码。
Code
写了个对拍没问题,姑且当它是对的吧
#include <bits/stdc++.h>
#define fo(i,a,b) for(int i=a;i<=b;++i)
#define fod(i,a,b) for(inti i=a;i>=b;--i)
#define N 200005
#define M 13000005
#define LL long long
#define mo 998244353
using namespace std;
int n,m,l,fs[N],nt[2*N],dt[2*N],m1;
vector<int> qs[N];
LL ksm(LL k,LL n)
{
LL s=1;
for(;n;n>>=1,k=k*k%mo) if(n&1) s=s*k%mo;
return s;
}
int n1,t[M][2],sz[M],rt[N];
LL sp[M],g[N],f[N],lz[M][2];
void nwp(int &k)
{
if(!k) k=++n1,lz[k][0]=1,lz[k][1]=0;
}
void ins(int k,int l,int r,int x,int v)
{
if(l==r) {sp[k]=0,sz[k]=1;return;}
int mid=(l+r)>>1;
if(x<=mid) nwp(t[k][0]),ins(t[k][0],l,mid,x,v);
else nwp(t[k][1]),ins(t[k][1],mid+1,r,x,v);
sp[k]=(sp[t[k][0]]+sp[t[k][1]]);
if(sp[k]>=mo) sp[k]-=mo;
sz[k]=sz[t[k][0]]+sz[t[k][1]];
}
LL gp,fp,fk,vs;
void upd(int k,LL u,LL v)
{
sp[k]=(u*sp[k]+v*sz[k])%mo;
lz[k][0]=lz[k][0]*u%mo;
lz[k][1]=(lz[k][1]*u%mo+v)%mo;
}
void down(int k)
{
if(lz[k][0]!=1||lz[k][1]!=0)
{
if(t[k][0]) upd(t[k][0],lz[k][0],lz[k][1]);
if(t[k][1]) upd(t[k][1],lz[k][0],lz[k][1]);
lz[k][0]=1,lz[k][1]=0;
}
}
void mrg(int &k,int x,int l,int r)
{
if(!k)
{
if(!x) return;
k=x,upd(k,mo-fk,gp*fk%mo);
return;
}
if(!x) {upd(k,(gp-fp+mo)%mo,0);return;}
if(l==r) {sp[k]=(gp-sp[x]+mo)%mo*sp[k]%mo,sz[k]=sz[k]|sz[x];return;}
int mid=(l+r)>>1;
down(k),down(x);
mrg(t[k][0],t[x][0],l,mid);
mrg(t[k][1],t[x][1],mid+1,r);
sp[k]=(sp[t[k][0]]+sp[t[k][1]])%mo;
sz[k]=sz[t[k][0]]+sz[t[k][1]];
}
void dfs(int k,int fa)
{
f[k]=1;
nwp(rt[k]);
int r=qs[k].size();
fo(j,0,r-1) ins(rt[k],1,m,qs[k][j],0);
for(int i=fs[k];i;i=nt[i])
{
int p=dt[i];
if(p!=fa)
{
dfs(p,k);
gp=g[p],fk=f[k],fp=f[p];
mrg(rt[k],rt[p],1,m);
f[k]=(g[p]-f[p]+mo)%mo*f[k]%mo;
}
}
g[k]=(f[k]*(LL)(m-sz[rt[k]])%mo+sp[rt[k]])%mo;
}
void link(int x,int y)
{
nt[++m1]=fs[x];
dt[fs[x]=m1]=y;
}
int main()
{
cin>>n>>m>>l;
fo(i,1,n-1)
{
int x,y;
scanf("%d%d",&x,&y);
link(x,y),link(y,x);
}
fo(i,1,l)
{
int x,y;
scanf("%d%d",&x,&y);
qs[x].push_back(y);
}
dfs(1,0);
printf("%lld\n",g[1]);
}
【PKUSC2019】树染色【线段树合并】【树形DP】的更多相关文章
- 【BZOJ5210】最大连通子块和 树剖线段树+动态DP
[BZOJ5210]最大连通子块和 Description 给出一棵n个点.以1为根的有根树,点有点权.要求支持如下两种操作: M x y:将点x的点权改为y: Q x:求以x为根的子树的最大连通子块 ...
- [CF1007D]Ants[2-SAT+树剖+线段树优化建图]
题意 我们用路径 \((u, v)\) 表示一棵树上从结点 \(u\) 到结点 \(v\) 的最短路径. 给定一棵由 \(n\) 个结点构成的树.你需要用 \(m\) 种不同的颜色为这棵树的树边染色, ...
- UVALive 7148 LRIP【树分治+线段树】
题意就是要求一棵树上的最长不下降序列,同时不下降序列的最小值与最大值不超过D. 做法是树分治+线段树,假设树根是x,y是其当前需要处理的子树,对于子树y,需要处理出两个数组MN,MX,MN[i]表示以 ...
- BZOJ4317Atm的树&BZOJ2051A Problem For Fun&BZOJ2117[2010国家集训队]Crash的旅游计划——二分答案+动态点分治(点分树套线段树/点分树+vector)
题目描述 Atm有一段时间在虐qtree的题目,于是,他满脑子都是tree,tree,tree…… 于是,一天晚上他梦到自己被关在了一个有根树中,每条路径都有边权,一个神秘的声音告诉他,每个点到其他的 ...
- dfs序+主席树 或者 树链剖分+主席树(没写) 或者 线段树套线段树 或者 线段树套splay 或者 线段树套树状数组 bzoj 4448
4448: [Scoi2015]情报传递 Time Limit: 20 Sec Memory Limit: 256 MBSubmit: 588 Solved: 308[Submit][Status ...
- 线段树简单入门 (含普通线段树, zkw线段树, 主席树)
线段树简单入门 递归版线段树 线段树的定义 线段树, 顾名思义, 就是每个节点表示一个区间. 线段树通常维护一些区间的值, 例如区间和. 比如, 上图 \([2, 5]\) 区间的和, 为以下区间的和 ...
- 【UOJ#388】【UNR#3】配对树(线段树,dsu on tree)
[UOJ#388][UNR#3]配对树(线段树,dsu on tree) 题面 UOJ 题解 考虑一个固定区间怎么计算答案,把这些点搞下来建树,然后\(dp\),不难发现一个点如果子树内能够匹配的话就 ...
- [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树)
[BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树) 题面 原题面有点歧义,不过从样例可以看出来真正的意思 有n个位置,每个位置可以看做一个集合. ...
- bzoj 3110 [Zjoi2013]K大数查询——线段树套线段树(标记永久化)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3110 第一道线段树套线段树! 第一道标记永久化! 为什么为什么写了两个半小时啊…… 本想线段 ...
- [BZOJ 1901] Dynamic Rankings 【树状数组套线段树 || 线段树套线段树】
题目链接:BZOJ - 1901 题目分析 树状数组套线段树或线段树套线段树都可以解决这道题. 第一层是区间,第二层是权值. 空间复杂度和时间复杂度均为 O(n log^2 n). 线段树比树状数组麻 ...
随机推荐
- 纯前端表格控件SpreadJS以专注业务、提升效率赢得用户与市场
提起华为2012实验室,你可能有点陌生. 但你一定还对前段时间华为的那封<海思总裁致员工的一封信>记忆犹新,就在那篇饱含深情的信中,我们知道了华为为确保公司大部分产品的战略安全和连续供应, ...
- ASP.NET Core中使用EF Core(MySql)Code First
⒈添加依赖 MySql.Data.EntityFrameworkCore ⒉在appsettings.json配置文件中配置数据库连接字符串 { "Logging": { &quo ...
- Python进阶编程 类与类的关系
类与类的关系 依赖关系 # 依赖关系: 将一个类的类名或者对象传给另一个类的方法中. class Elephant: def __init__(self, name): self.name = nam ...
- Hive 教程(九)-python with hive
本文介绍用 python 远程连接 hive,此时需要 hive 启动 hiveserver2 服务 windows 下报如下错误 thrift.transport.TTransport.TTrans ...
- JVM(6)之 二次标记
开发十年,就只剩下这套架构体系了! >>> 上一章我们讲到了标记,但是不是被标记了就肯定会被回收呢?不知道小伙伴们记不记得Object类有一个finalize()方法,所有类都继 ...
- Vue路由守卫之路由独享守卫
路由独立守卫,顾名思义就是这个路由自己的守卫任务,就如同咱们LOL,我们守卫的就是独立一条路,保证我们这条路不要被敌人攻克(当然我们也得打团配合) 在官方定义是这样说的:你可以在路由配置上直接定义 ...
- java中的Enum在@RestController(@ResponseBody) 注解下返回的表现
参考文档 枚举 public enum CouponType { PLATFORM("平台优惠券"), NEWCOMER("新人专享优惠券"), INVITE( ...
- Spring Cloud,Docker
Spring Cloud 先决条件 Spring cloud 基于spring boot,spring,java Spring Cloud解决的问题 分布式微服务架构和微服务监控.注册于发现.跟踪等一 ...
- pytorch简单框架
网络搭建: mynn.py: import torchfrom torch import nnclass mynn(nn.Module): def __init__(self): super(mynn ...
- 在mysql 上如何在不影响生产的情况下删除一个大表
mysql 中常用的删除的方法基本上有下面三种方式: 1.delete 一般用于删除少量表中的数据 优化建议,一定要加上where 条件,并且where条件的列上 一定要有主键或者索引.否则会出现全表 ...