【GDOI 2016 Day1】疯狂动物城
题目
分析
注意注意:码农题一道,打之前做好心理准备。
对于操作1、2,修改或查询x到y的路径,显然树链剖分。
对于操作2,我们将x到y的路径分为x到lca(x,y)和lca(x,y)到y两部分。
对于第一部分的某个点i,设它到y的距离为s,那么s=deep[i]+deep[y]-2*deep[lca(x,y)],i对答案的贡献为a[i]s(s+1)/2,如果不考虑除以2,设t=deep[y]-2*deep[lca(x,y)],则贡献为a[i]*deep[i]2+a[i]*deep[i]*(2*t+1)+a[i]*(t+t2)。
对于第二部分的点i,s=deep[y]-deep[i],设lca(x,y),则贡献为a[i]*deep[i]2-a[i]*deep[i]*(2*t+1)+a[i]*(t+t2)。
接着,对于每个点i我们就用线段树来维护a[i]*deep[i]^2、a[i]*deep[i]以及a[i]。
注意:最后记得除2,由于mod 20160501,用逆元。还有,lca(x,y)的贡献会重复,要减掉重复的。
对于操作3,打个可持久化线段树。
#include <cmath>
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <queue>
const long long maxlongint=2147483647;
const long long mo=20160501;
using namespace std;
struct trees
{
long long lazy,l,r;
long long v1;//a[i]*deep[i]^2
long long v2;//a[i]*deep[i]
long long v3;//a[i]
long long d;
long long d2;
}tree[7000000];
long long g[200005][25],d[200005],son[200005],deep[200005],size[200005],fa[200005],top[200005],bef[200005],f[220005];
long long last[200005],next[200005],to[200005];
long long n,m,ans,t,po,tot,tt;
long long bj(long long x,long long y)
{
next[++tot]=last[x];
last[x]=tot;
to[tot]=y;
}
long long premi()
{
for(long long j=1;j<=log2(n);j++)
{
for(long long i=1;i<=n;i++)
{
g[i][j]=g[g[i][j-1]][j-1];
}
}
}
long long build(long long x)
{
size[x]=1;
g[x][0]=fa[x];
long long mx=0;
for(long long i=last[x];i;i=next[i])
{
long long j=to[i];
if(j!=fa[x])
{
fa[j]=x;
deep[j]=deep[x]+1;
build(j);
size[x]+=size[j];
if(size[j]>mx)
{
mx=size[j];
son[x]=j;
}
}
}
}
long long build1(long long x)
{
d[++tot]=x;
bef[x]=tot;
if(!top[x])
{
top[x]=x;
}
if(son[x])
{
top[son[x]]=top[x];
build1(son[x]);
}
for(long long i=last[x];i;i=next[i])
{
long long j=to[i];
if(j!=fa[x] && son[x]!=j)
{
build1(j);
}
}
}
long long bnew(long long v,long long l,long long r,long long e)
{
if(l==r)
{
tree[v].lazy=0;
return 0;
}
long long mid=(l+r)/2;
if(e!=2)
{
tree[++tot]=tree[tree[v].l];
if(e==3)
{
tree[tot].v1=(tree[tot].v1+tree[v].lazy*tree[tot].d2)%mo;
tree[tot].v2=(tree[tot].v2+tree[v].lazy*tree[tot].d)%mo;
tree[tot].v3=(tree[tot].v3+tree[v].lazy*(mid-l+1))%mo;
tree[tot].lazy=(tree[tot].lazy+tree[v].lazy)%mo;
}
tree[v].l=tot;
}
if(e>=2)
{
tree[++tot]=tree[tree[v].r];
if(e==3)
{
tree[tot].v1=(tree[tot].v1+tree[v].lazy*tree[tot].d2)%mo;
tree[tot].v2=(tree[tot].v2+tree[v].lazy*tree[tot].d)%mo;
tree[tot].v3=(tree[tot].v3+tree[v].lazy*(r-(mid+1)+1))%mo;
tree[tot].lazy=(tree[tot].lazy+tree[v].lazy)%mo;
}
tree[v].r=tot;
}
tree[v].lazy=0;
}
long long change1(long long v,long long l,long long r,long long x,long long value)
{
if(l==r)
{
tree[v].v3=value;
tree[v].v2=value*deep[d[l]];
tree[v].v1=value*deep[d[l]]*deep[d[l]];
tree[v].d=deep[d[l]];
tree[v].d2=deep[d[l]]*deep[d[l]];
return 0;
}
long long mid=(l+r)/2;
if(x<=mid)
{
if(!tree[v].l) tree[v].l=++tot;
change1(tree[v].l,l,mid,x,value);
}
else
{
if(!tree[v].r) tree[v].r=++tot;
change1(tree[v].r,mid+1,r,x,value);
}
tree[v].v1=tree[tree[v].l].v1+tree[tree[v].r].v1;
tree[v].v2=tree[tree[v].l].v2+tree[tree[v].r].v2;
tree[v].v3=tree[tree[v].l].v3+tree[tree[v].r].v3;
tree[v].d=tree[tree[v].l].d+tree[tree[v].r].d;
tree[v].d2=tree[tree[v].l].d2+tree[tree[v].r].d2;
}
long long change(long long v,long long l,long long r,long long x,long long y,long long value)
{
if(l==x && r==y)
{
tree[v].v1=tree[v].v1+value*tree[v].d2;
tree[v].v2=tree[v].v2+value*tree[v].d;
tree[v].v3=tree[v].v3+value*(r-l+1);
tree[v].lazy=tree[v].lazy+value;
return 0;
}
bool bz=true;
if(tree[v].lazy)
{
bnew(v,l,r,3);
bz=false;
}
long long mid=(l+r)/2;
if(y<=mid)
{
if(bz) bnew(v,l,r,1);
change(tree[v].l,l,mid,x,y,value);
}
else
if(x>=mid+1)
{
if(bz) bnew(v,l,r,2);
change(tree[v].r,mid+1,r,x,y,value);
}
else
{
if(bz) bnew(v,l,r,4);
change(tree[v].l,l,mid,x,mid,value);
change(tree[v].r,mid+1,r,mid+1,y,value);
}
tree[v].v1=tree[tree[v].l].v1+tree[tree[v].r].v1;
tree[v].v2=tree[tree[v].l].v2+tree[tree[v].r].v2;
tree[v].v3=tree[tree[v].l].v3+tree[tree[v].r].v3;
}
long long lca(long long x,long long y)
{
if(deep[x]>deep[y])
{
x=x^y;
y=x^y;
x=x^y;
}
for(long long i=log2(n);i>=0;i--)
{
if(deep[g[y][i]]>deep[x])
y=g[y][i];
}
if(deep[y]!=deep[x]) y=g[y][0];
for(long long i=log2(n);i>=0;i--)
{
if(g[y][i]!=g[x][i])
{
y=g[y][i];
x=g[x][i];
}
}
if(x!=y) y=g[y][0];
return y;
}
long long find(long long v,long long l,long long r,long long x,long long y,long long value,long long o)
{
if(l==x && r==y)
{
return ((tree[v].v1+o*tree[v].v2*(2*value+1)+tree[v].v3*(value+value*value))%mo+mo)%mo;
}
if(tree[v].lazy)
{
bnew(v,l,r,3);
}
long long mid=(l+r)/2,e=0;
if(y<=mid)
{
e=find(tree[v].l,l,mid,x,y,value,o);
}
else
if(x>=mid+1)
{
e=find(tree[v].r,mid+1,r,x,y,value,o);
}
else
{
e=find(tree[v].l,l,mid,x,mid,value,o)+find(tree[v].r,mid+1,r,mid+1,y,value,o);
}
tree[v].v1=tree[tree[v].l].v1+tree[tree[v].r].v1;
tree[v].v2=tree[tree[v].l].v2+tree[tree[v].r].v2;
tree[v].v3=tree[tree[v].l].v3+tree[tree[v].r].v3;
return e;
}
long long work(long long x,long long y,long long z)
{
if(z)
{
f[++t]=++tot;
tree[tot]=tree[f[tt]];
tt=t;
while(top[x]!=top[y])
{
if(deep[top[x]]>=deep[top[y]])
{
change(f[t],1,n,bef[top[x]],bef[x],z);
x=fa[top[x]];
}
else
{
change(f[t],1,n,bef[top[y]],bef[y],z);
y=fa[top[y]];
}
}
if(deep[x]>=deep[y])
{
change(f[t],1,n,bef[y],bef[x],z);
}
else
{
change(f[t],1,n,bef[x],bef[y],z);
}
}
else
{
ans=0;
long long lc=lca(x,y),xx=x,yy=y;
long long p=deep[yy]-deep[lc]*2;
x=xx;
y=lc;
if(x!=y)
{
while(top[x]!=top[y])
{
if(deep[top[x]]>=deep[top[y]])
{
ans=((ans+find(f[tt],1,n,bef[top[x]],bef[x],p,1))%mo+mo)%mo;
x=fa[top[x]];
}
else
{
ans=((ans+find(f[tt],1,n,bef[top[y]],bef[y],p,1))%mo+mo)%mo;
y=fa[top[y]];
}
}
if(deep[x]>=deep[y])
{
ans=((ans+find(f[tt],1,n,bef[y],bef[x],p,1))%mo+mo)%mo;
}
else
{
ans=((ans+find(f[tt],1,n,bef[x],bef[y],p,1))%mo+mo)%mo;
}
if(lc!=yy) ans=((ans-find(f[tt],1,n,bef[lc],bef[lc],p,1))%mo+mo)%mo;
}
x=lc;
y=yy;
p=deep[yy];
if(x!=y)
{
while(top[x]!=top[y])
{
if(deep[top[x]]>=deep[top[y]])
{
ans=((ans+find(f[tt],1,n,bef[top[x]],bef[x],p,-1))%mo+mo)%mo;
x=fa[top[x]];
}
else
{
ans=((ans+find(f[tt],1,n,bef[top[y]],bef[y],p,-1))%mo+mo)%mo;
y=fa[top[y]];
}
}
if(deep[x]>=deep[y])
{
ans=((ans+find(f[tt],1,n,bef[y],bef[x],p,-1))%mo+mo)%mo;
}
else
{
ans=((ans+find(f[tt],1,n,bef[x],bef[y],p,-1))%mo+mo)%mo;
}
}
ans=(ans%mo+mo)*10080251%mo;
printf("%lld\n",ans);
}
}
int main()
{
scanf("%lld%lld",&n,&m);
for(long long i=1;i<=n-1;i++)
{
long long x,y;
scanf("%lld%lld",&x,&y);
bj(x,y);
bj(y,x);
}
tot=0;
deep[1]=1;
build(1);
build1(1);
tot=1;
f[0]=1;
t=0;
tt=0;
for(long long i=1;i<=n;i++)
{
long long x;
scanf("%lld",&x);
change1(1,1,n,bef[i],x);
}
premi();
ans=0;
for(long long i=1;i<=m;i++)
{
long long p,x,y,z;
scanf("%lld",&p);
if(p==1)
{
scanf("%lld%lld%lld",&x,&y,&z);
work(x^ans,y^ans,z);
}
else
if(p==2)
{
scanf("%lld%lld",&x,&y);
work(x^ans,y^ans,0);
}
else
{
scanf("%lld",&x);
tt=x^ans;
}
}
}
【GDOI 2016 Day1】疯狂动物城的更多相关文章
- 《疯狂动物城》主题曲《TryEverything》中文翻译
<疯狂动物城>主题曲<TryEverything>夏奇拉激情献唱,很多事情是需要是试试,不试试就不知道可以成功. Oh oh oh oh oooh 哦哦哦哦哦 Oh oh oh ...
- [GDOI2016][树链剖分+主席树]疯狂动物城
题面 Description Nick 是只在动物城以坑蒙拐骗为生的狐狸,儿时受到偏见的伤害,放弃了自己的理想.他被兔子 Judy 设下圈套,被迫与她合作查案,而卷入意想不到的阴谋,历尽艰险后成为搭档 ...
- COGS-2049 疯狂动物城
Description 你意外来到了一个未知的星球, 这里是一个动物乌托邦, 生活着一群拥有非凡智力的动物. 你遇到了一个叫做尼克的狐狸, 他准备给他的 GF 过生日 . 他将制作一个巨大的多层蛋糕, ...
- 【xsy2440】【GDOI2016】疯狂动物城
感受一下这恐怖的题目长度~~~ 其实题意很裸,但是作为GDOI的一道防AK题,自然有他优秀的地方. 简化题意:给出一棵树,要求支持三个操作: 1.修改点值 2.询问点$x$到$y$之间的一些东东 3. ...
- 【GDOI 2016 Day1】第二题 最长公共子串
分析 首先,可以发现,区间是可以合并滴.把区间按左端点排序,对于两个区间[l1,r1].[l2,r2],当l1<=l2 and r1>=l2,那么,将它们合成一个新的区间[l1,r2].当 ...
- https://docs.huihoo.com/infoq/qconbeijing/2016/day1/工程效率提升专题/2-2-支持百度万人开发的工具装备及百度工程能力建设-夏仲璞.pdf
https://docs.huihoo.com/infoq/qconbeijing/2016/day1/工程效率提升专题/2-2-支持百度万人开发的工具装备及百度工程能力建设-夏仲璞.pdf http ...
- GDOI 2016 & APIO 2016 游记
缓慢施工中...... UPD:APIO游记已烂尾......因为Cu滚粗+生病一直没心情写..过了几天就发现APIO的事都快忘光了...去看KPM的就可以啦 今年apio竟然没和gdoi撞...智障 ...
- Noip 2016 Day1 题解
老师让我们刷历年真题, 然后漫不经心的说了一句:“你们就先做做noip2016 day1 吧” ...... 我还能说什么,,,,,老师你这是明摆着伤害我们啊2333333333 预计分数:100+2 ...
- noip 2016 day1 T1玩具谜题
题目描述 小南有一套可爱的玩具小人, 它们各有不同的职业. 有一天, 这些玩具小人把小南的眼镜藏了起来. 小南发现玩具小人们围成了一个圈,它们有的面朝圈内,有的面朝圈外.如下图: 这时singer告诉 ...
随机推荐
- 2018 icpc 徐州
A 矩阵树定理可以用于最小生成树计数,最直观的做法就是求个mst,再用矩阵树定理求最小生成树个数,但是n<=1e5,显然不是o(n^3)可以做出来的. 考虑随机数据生成器,固定1e5的边,但是边 ...
- Spring MVC静态资源处理——<mvc:resources /> ||<mvc:default-servlet-handler />
优雅REST风格的资源URL不希望带 .html 或 .do 等后缀.由于早期的Spring MVC不能很好地处理静态资源,所以在web.xml中配置DispatcherServlet的请求映射,往往 ...
- STM32 USB开发(三) 基于F105RBT6核心板开发的自定义HID收发(FS)
硬件设计 该核心板的USB插口有两个,一个是用于USB Slave的,可以用来做HID设备,把模拟STM32模拟为U盘等:另一个是USB Host设备,可以对插上的U盘的数据进行读写. 图中J2是Mi ...
- CNN文本分类
CNN用于文本分类本就是一个不完美的解决方案,因为CNN要求输入都是一定长度的,而对于文本分类问题,文本序列是不定长的,RNN可以完美解决序列不定长问题, 因为RNN不要求输入是一定长度的.那么对于C ...
- RabbitMQ 安装步骤
RabbitMQ安装步骤 一.安装erlang 1.下载erlang wget https://packages.erlang-solutions.com/erlang-solutions-1.0-1 ...
- 蚁群算法解决TSP问题
代码实现 运行结果及参数展示 alpha=1beta=5 rho=0.1 alpha=1beta=1rho=0.1 alpha=0.5beta=1rho=0.1 概念蚁群算法(AG)是一种模拟蚂蚁觅 ...
- MySQL-快速入门(1)基本数据库、表操作语句
1.创建数据库 create database db_name;show create database db_name\G; //查看数据创建语句show databases; //查看当前创建的数 ...
- HNUSTOJ 1604:Operations
1604: Operations 时间限制: 2 Sec 内存限制: 128 MB 提交: 313 解决: 97 [提交][状态][讨论版] 题目描述 You can perform the fo ...
- MySQL时间戳加减转日期
一.时间戳计算前N天后N天并转化为日期,本例是将某个时间戳转为日期,并计算出与该日期前后相差7天的日期: , , , DAY)as 'after'; 输出: # before, now, after ...
- application详解
Application对象是HttpApplicationState类的一个实例,Application状态是整个应用程序全局的.本文主要详细介绍Application对象的用法. 一.全局应用程序类 ...