传送门

模板集合吧,除了码农,没啥难的。。。

bzoj2243:[SDOI2011]染色十分相像,但是多了点集和查询的区别

然后点集显然可以看出是虚树问题,查询可以用点分治\(O(nlogn)\),也可以二次换根\(O(n)\),显然二次换根\(O(n)\)更加优秀

然后码就是了

代码:

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
using namespace std;
void read(int &x) {
char ch; bool ok;
for(ok=0,ch=getchar(); !isdigit(ch); ch=getchar()) if(ch=='-') ok=1;
for(x=0; isdigit(ch); x=x*10+ch-'0',ch=getchar()); if(ok) x=-x;
}
#define rg register
const int maxn=1e5+10;
int n,m,c[maxn],dep[maxn],st[maxn],top[maxn],id[maxn],tmp,f[maxn][20];
int nid[maxn],size[maxn],d[maxn],tot,w[maxn],g[maxn],ans[maxn],ssize[maxn];
struct oo{
int pre[maxn*2],nxt[maxn*2],h[maxn],cnt;
void add(int x,int y)
{
pre[++cnt]=y,nxt[cnt]=h[x],h[x]=cnt;
pre[++cnt]=x,nxt[cnt]=h[y],h[y]=cnt;
}
}a,b;
struct segment_tree{int l,r,sum,ls,rs,cov;}s[maxn*4];
void update(int x)
{
s[x].sum=s[x<<1].sum+s[x<<1|1].sum-(s[x<<1].rs==s[x<<1|1].ls);
s[x].ls=s[x<<1].ls,s[x].rs=s[x<<1|1].rs;
}
void build(int x,int l,int r)
{
s[x].l=l,s[x].r=r;int mid=(l+r)>>1;
if(l==r){s[x].sum=1,s[x].ls=s[x].rs=c[nid[l]];return ;}
build(x<<1,l,mid),build(x<<1|1,mid+1,r);
update(x);
}
void dfs(int x)
{
size[x]=1;
for(rg int i=1;i<20;i++)
{
if(dep[x]<(1<<i))break;
f[x][i]=f[f[x][i-1]][i-1];
}
for(rg int i=a.h[x];i;i=a.nxt[i])
if(a.pre[i]!=f[x][0])
{
f[a.pre[i]][0]=x,dep[a.pre[i]]=dep[x]+1;
dfs(a.pre[i]),size[x]+=size[a.pre[i]];
}
}
void dfs1(int x,int f)
{
top[x]=f,id[x]=++tmp,nid[tmp]=x;int k=0;
for(int i=a.h[x];i;i=a.nxt[i])
if(dep[a.pre[i]]>dep[x]&&size[a.pre[i]]>size[k])k=a.pre[i];
if(!k)return ;dfs1(k,f);
for(rg int i=a.h[x];i;i=a.nxt[i])
if(dep[a.pre[i]]>dep[x]&&a.pre[i]!=k)dfs1(a.pre[i],a.pre[i]);
}
int lca(int x,int y)
{
if(dep[x]>dep[y])swap(x,y);
int poor=dep[y]-dep[x];
for(rg int i=19;i>=0;i--)if(poor&(1<<i))y=f[y][i];
if(x==y)return x;
for(rg int i=19;i>=0;i--)if(f[x][i]!=f[y][i])x=f[x][i],y=f[y][i];
return x==y?x:f[x][0];
}
void pushdown(int x)
{
s[x<<1].sum=s[x<<1|1].sum=1;
s[x<<1].cov=s[x<<1|1].cov=s[x].cov;
s[x<<1].ls=s[x<<1].rs=s[x].cov;
s[x<<1|1].ls=s[x<<1|1].rs=s[x].cov;
s[x].cov=0;
}
void change(int x,int l,int r,int v)
{
if(l<=s[x].l&&r>=s[x].r)
{
s[x].cov=s[x].ls=s[x].rs=v,s[x].sum=1;
return ;
}
if(s[x].cov)pushdown(x);
int mid=(s[x].l+s[x].r)>>1;
if(l<=mid)change(x<<1,l,r,v);
if(r>mid)change(x<<1|1,l,r,v);
update(x);
}
void modify(int x,int y,int z)
{
while(top[x]!=top[y])
{
if(dep[top[x]]<dep[top[y]])swap(x,y);
change(1,id[top[x]],id[x],z),x=f[top[x]][0];
}
if(id[x]>id[y])swap(x,y);
change(1,id[x],id[y],z);
}
bool cmp(int x,int y){return id[x]<id[y];}
segment_tree get(int x,int l,int r)
{
if(l<=s[x].l&&r>=s[x].r)return s[x];
int mid=(s[x].l+s[x].r)>>1;
if(s[x].cov)pushdown(x);
if(r<=mid)return get(x<<1,l,r);
else if(l>mid)return get(x<<1|1,l,r);
else
{
segment_tree a=get(x<<1,l,r),b=get(x<<1|1,l,r),c;
c.sum=a.sum+b.sum-(a.rs==b.ls);
c.ls=a.ls,c.rs=b.rs;
return c;
}
}
int qsum(int x,int y)
{
int sum=0;segment_tree a,b;
while(top[x]!=top[y])
{
if(dep[top[x]]<dep[top[y]])swap(x,y);
a=get(1,id[top[x]],id[x]);
x=f[top[x]][0],b=get(1,id[x],id[x]);
sum=sum+a.sum-(b.rs==a.ls);
}
if(id[x]>id[y])swap(x,y);
a=get(1,id[x],id[y]),sum+=a.sum;
return sum;
}
void prepare(int x,int fa)
{
ssize[x]=0;if(w[x])ssize[x]++;
for(rg int i=b.h[x];i;i=b.nxt[i])
if(b.pre[i]!=fa)prepare(b.pre[i],x),ssize[x]+=ssize[b.pre[i]];
}
void solve(int x,int fa,int tot)
{
if(w[x])ans[w[x]]=g[x];
for(rg int i=b.h[x];i;i=b.nxt[i])
if(b.pre[i]!=fa)
{
int now=qsum(x,b.pre[i])-1;
g[b.pre[i]]=g[x]+now*(tot-2*ssize[b.pre[i]]);
solve(b.pre[i],x,tot);
}
}
int main()
{
read(n),read(m);
for(rg int i=1;i<=n;i++)read(c[i]);
for(rg int i=1,x,y;i<n;i++)read(x),read(y),a.add(x,y);
dfs(1),dfs1(1,1),build(1,1,n);
for(rg int i=1,q,x,y,z;i<=m;i++)
{
read(q);
if(q==1)read(x),read(y),read(z),modify(x,y,z);
else
{
read(z);tot=0;b.cnt=0;memset(b.h,0,sizeof b.h);
for(rg int j=1;j<=z;j++)read(d[j]),w[d[j]]=j;
sort(d+1,d+z+1,cmp),st[++tot]=1;
for(rg int j=1;j<=z;j++)
{
int w=0,e=0;
while(tot&&lca(st[tot],d[j])!=st[tot])
{
if(w)b.add(st[tot],w);
w=st[tot],tot--;
}
if(w)e=lca(w,d[j]),b.add(w,e);
if(e&&e!=st[tot])st[++tot]=e;
st[++tot]=d[j];
}
while(tot>1)b.add(st[tot],st[tot-1]),tot--;
for(rg int j=1;j<=z;j++)g[1]=g[1]+qsum(1,d[j]);
prepare(1,0),solve(1,0,z);
for(rg int j=1;j<=z;j++)printf("%d ",ans[j]);printf("\n");
for(rg int j=1;j<=z;j++)w[d[j]]=0,g[j]=0;
}
}
}

luoguP4242树上的毒瘤的更多相关文章

  1. [luogu4242] 树上的毒瘤

    题目描述 这棵树上有n个节点,由n−1条树枝相连.初始时树上都挂了一个毒瘤,颜色为ci.接下来Salamander将会进行q个操作. Salamander有时会修改树上某个点到另外一个点的简单路径上所 ...

  2. 洛谷P4242 树上的毒瘤

    解:首先有个套路是一条边的权值是[两端点颜色不同].这个用树剖直接维护,支持修改. 每次询问建虚树,查询虚树上每条边的权值.然后树形DP,用开店的方法,每个点链加链查. #include <bi ...

  3. Noip前的大抱佛脚----赛前任务

    赛前任务 tags:任务清单 前言 现在xzy太弱了,而且他最近越来越弱了,天天被爆踩,天天被爆踩 题单不会在作业部落发布,所以可(yi)能(ding)会不及时更新 省选前的练习莫名其妙地成为了Noi ...

  4. 洛谷P4426 毒瘤 [HNOI/AHOI2018] 虚树+树上dp

    正解:虚树+树上dp 解题报告: 传送门! 首先解释一下题意趴,,,语文70pts选手已经开始看不懂题辣QAQ 大概就是个给一个图,求独立集方案,且保证图是联通的,边的数量最多只比点多10 首先思考如 ...

  5. 【HNOI 2018】毒瘤

    Problem Description 从前有一名毒瘤. 毒瘤最近发现了量产毒瘤题的奥秘.考虑如下类型的数据结构题:给出一个数组,要求支持若干种奇奇怪怪的修改操作(例如给一个区间内的数同时加上 \(c ...

  6. 「HNOI2016」数据结构大毒瘤

    真是 \(6\) 道数据结构毒瘤... 开始口胡各种做法... 「HNOI2016」网络 整体二分+树状数组. 开始想了一个大常数 \(O(n\log^2 n)\) 做法,然后就被卡掉了... 发现直 ...

  7. 【BZOJ5287】[HNOI2018]毒瘤(动态规划,容斥)

    [BZOJ5287][HNOI2018]毒瘤(动态规划,容斥) 题面 BZOJ 洛谷 题解 考场上想到的暴力做法是容斥: 因为\(m-n\le 10\),所以最多会多出来\(11\)条非树边. 如果就 ...

  8. [Codeforces743D][luogu CF743D]Chloe and pleasant prizes[树状DP入门][毒瘤数据]

    这个题的数据真的很毒瘤,身为一个交了8遍的蒟蒻的呐喊(嘤嘤嘤) 个人认为作为一个树状DP的入门题十分合适,同时建议做完这个题之后再去做一下这个题 选课 同时在这里挂一个选取节点型树形DP的状态转移方程 ...

  9. 「HNOI2018」毒瘤

    「HNOI2018」毒瘤 解题思路 先考虑只有一棵树的情况,经典独立集计数. \[ dp[u][0]=\prod (dp[v][0]+dp[v][1]) \\ dp[u][1]=\prod dp[v] ...

随机推荐

  1. JS中正则匹配开头不带空格,结尾也不带空格的字符串

    在做项目的时候,要求限制SSID的长度.以及开头和结尾不能是空格. var reg = /^\S.{0,30}\S$/ "$$$  $$".match(reg);   ==> ...

  2. 时间操作(Java版)—获取给定日期N天后的日期

    版权声明:本文为博主原创文章.未经博主同意不得转载. https://blog.csdn.net/wangshuxuncom/article/details/34896777         获取给定 ...

  3. linux mount的安装配置

    1.首先需要安装nfs-utils 和 portmap: yum install nfs-utils portmap (安装软件,也可直接安装 yum install nfs* portmap)chk ...

  4. MongoDB 倾向于将数据都放在一个 Collection 下吗?

    不是这样的. Collection 的单个 doc 有大小上限,现在是 16MB,这就使得你不可能把所有东西都揉到一个 collection 里.而且如果 collection 结构过于复杂,既会影响 ...

  5. leeetcode 735. Asteroid Collision

    We are given an array asteroids of integers representing asteroids in a row. For each asteroid, the ...

  6. Android Weekly Notes Issue #242

    Android Weekly Issue #242 January 29th, 2017 Android Weekly Issue #242 本期内容包括: Android中常用的设计模式; 基于No ...

  7. PHP截取中英文混合字符

    <?php //////////////////////////////////////////////////////////////////// // PHP截取中英文及标点符号混合的字符串 ...

  8. 基于logstash+elasticsearch+kibana的日志收集分析方案(Windows)

    一 方案背景     通常,日志被分散的储存不同的设备上.如果你管理数十上百台服务器,你还在使用依次登录每台机器的传统方法查阅日志.这样是不是感觉很繁琐和效率低下.开源实时日志分析ELK平台能够完美的 ...

  9. Myeclipse项目内容没有报错但是项目上面却有红色叉叉

    当src文件夹为空的时候,git是不提交空文件夹的,所以check出来的项目中没有src文件夹,这个时候也会出现此问题.

  10. 【MongoDB学习-在.NET中的简单操作类】

    1.新建MVC项目, 管理NuGet包,进入下载MongDB.net库文件 2.新增项目DAL数据访问层,引用以下库文件: 3.C# 访问MongoDB通用方法类: using MongoDB.Dri ...