51nod-1462: 树据结构
【传送门:51nod-1462】
简要题意:
给出一棵n个点的树,每个点有两个权值v,t
有Q个操作,有两种操作:
1.将x到根上的路径上的点的v值都加上d
2.将x到根上的路径上的点的t值都加上每个点的v值*d
最后求出所有点的t值
题解:
显然可以直接树链剖分做,不过lazy标记下放真麻烦,因为操作有互相影响
发现一种神标记方法——用矩阵
对于一个点,它的信息表示为$$ \begin{matrix} 1 & 0 & 0 \\ 0 & 1 & 0 \\ v & t & 1 \\ \end{matrix} $$
而对于第一种操作,就将点的信息乘上$$ \begin{matrix} 1 & 0 & 0 \\ 0 & 1 & 0 \\ d & 0 & 1 \\ \end{matrix} $$
而第二种操作则乘上$$ \begin{matrix} 1 & d & 0 \\ 0 & 1 & 0 \\ 0 & 0 & 1 \\ \end{matrix} $$
这样就不用担心互相影响了,因为矩阵乘法满足结合律,所以直接将操作按顺序乘起来就行了
听说可以离线CDQ做
参考代码:
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<algorithm>
using namespace std;
typedef long long LL;
struct trnode
{
int l,r,lc,rc;
}tr[];int trlen;
struct Matrix
{
LL a[][];
Matrix()
{
memset(a,,sizeof(a));
}
}D[],cmp;
Matrix multi(Matrix a,Matrix b)
{
Matrix c;
for(int i=;i<=;i++)
{
for(int j=;j<=;j++)
{
for(int k=;k<=;k++)
{
c.a[i][j]+=a.a[i][k]*b.a[k][j];
}
}
}
return c;
}
void bt(int l,int r)
{
trlen++;int now=trlen;
tr[now].l=l;tr[now].r=r;
D[now].a[][]=D[now].a[][]=D[now].a[][]=;
tr[now].lc=tr[now].rc=-;
if(l<r)
{
int mid=(tr[now].l+tr[now].r)/;
tr[now].lc=trlen+;bt(l,mid);
tr[now].rc=trlen+,bt(mid+,r);
}
}
struct node
{
int x,y,next;
}a[];int len,last[];
void ins(int x,int y){a[++len]=(node){x,y,last[x]};last[x]=len;}
int son[],tot[];
void dfs1(int x)
{
tot[x]=;son[x]=;
for(int k=last[x];k;k=a[k].next)
{
int y=a[k].y;
dfs1(y);
if(tot[y]>tot[son[x]]) son[x]=y;
tot[x]+=tot[y];
}
}
int top[],ys[],z;
void dfs2(int x,int tp)
{
ys[x]=++z;top[x]=tp;
if(son[x]!=) dfs2(son[x],tp);
for(int k=last[x];k;k=a[k].next)
{
int y=a[k].y;
if(y!=son[x]) dfs2(y,y);
}
}
void update(int now)
{
int lc=tr[now].lc,rc=tr[now].rc;
if(lc!=-) D[lc]=multi(D[lc],D[now]);
if(rc!=-) D[rc]=multi(D[rc],D[now]);
memset(D[now].a,,sizeof(D[now].a));
D[now].a[][]=D[now].a[][]=D[now].a[][]=;
}
void change(int now,int l,int r)
{
if(tr[now].l==l&&tr[now].r==r)
{
D[now]=multi(D[now],cmp);
return ;
}
int lc=tr[now].lc,rc=tr[now].rc,mid=(tr[now].l+tr[now].r)/;
update(now);
if(r<=mid) change(lc,l,r);
else if(l>mid) change(rc,l,r);
else change(lc,l,mid),change(rc,mid+,r);
}
int fa[];
void solve(int x)
{
int tx=top[x];
while(x!=)
{
change(,ys[tx],ys[x]);
x=fa[tx];tx=top[x];
}
}
LL d[];
void out(int now)
{
if(tr[now].l==tr[now].r)
{
d[tr[now].l]=D[now].a[][];
return ;
}
int lc=tr[now].lc,rc=tr[now].rc;
update(now);
if(lc!=-) out(lc);
if(rc!=-) out(rc);
}
int main()
{
int n;
scanf("%d",&n);
len=;memset(last,,sizeof(last));
for(int i=;i<=n;i++)
{
scanf("%d",&fa[i]);
ins(fa[i],i);
}
dfs1();
z=;dfs2(,);
trlen=;bt(,z);
cmp.a[][]=cmp.a[][]=cmp.a[][]=;
int Q;
scanf("%d",&Q);
while(Q--)
{
int t,x;LL d;
scanf("%d%d%lld",&t,&x,&d);
if(t==) cmp.a[][]=,cmp.a[][]=d,solve(x);
else cmp.a[][]=,cmp.a[][]=d,solve(x);
}
out();
for(int i=;i<=n;i++) printf("%lld\n",d[ys[i]]);
return ;
}
51nod-1462: 树据结构的更多相关文章
- 51nod 1462 树据结构 | 树链剖分 矩阵乘法
题目链接 51nod 1462 题目描述 给一颗以1为根的树. 每个点有两个权值:vi, ti,一开始全部是零. Q次操作: 读入o, u, d o = 1 对u到根上所有点的vi += d o = ...
- 树状结构Java模型、层级关系Java模型、上下级关系Java模型与html页面展示
树状结构Java模型.层级关系Java模型.上下级关系Java模型与html页面展示 一.业务原型:公司的组织结构.传销关系网 二.数据库模型 很简单,创建 id 与 pid 关系即可.(pid:pa ...
- 分享使用NPOI导出Excel树状结构的数据,如部门用户菜单权限
大家都知道使用NPOI导出Excel格式数据 很简单,网上一搜,到处都有示例代码. 因为工作的关系,经常会有处理各种数据库数据的场景,其中处理Excel 数据导出,以备客户人员确认数据,场景很常见. ...
- 由简入繁实现Jquery树状结构
在项目中,我们经常会需要一些树状结构的样式来显示层级结构等,比如下图的样式,之前在学.net的时候可以直接拖个服务端控件过来直接使用非常方便.但是利用Jquery的一些插件,也是可以实现这些效果的,比 ...
- php实现树状结构无级分类
php实现树状结构无级分类 ).",'树2-1-1-2')";mysql_query($sql);?>
- Android无限级树状结构
通过对ListView简单的扩展.再封装,即可实现无限层级的树控件TreeView. package cn.asiontang.nleveltreelistview; import android.a ...
- dzzoffice的树型结构用户管理设计
在DzzOffice1.1的开发中,针对用户使用群体重新设计了,机构.部门.用户管理应用. 传统OA,企业相关程序,一般是设置机构-设置部门-设置职位-添加用户这样的步骤.每个步骤分为不同的管理界面. ...
- 使用Map辅助拼装树状结构,消除递归调用
目前菜单或其他树状结构在数据库中的存储,多数是以一个parentid作为关联字段,以一维形式存储.使用时全部查询出来,然后在内存中拼装成树状结构.现在主要涉及的是拼装方法的问题. 一般可以进行 递归调 ...
- lua 怎样输出树状结构的table?
为了让游戏前端数据输出更加条理,做了一个简单树状结构来打印数据. ccmlog.lua local function __tostring(value, indent, vmap) local str ...
- js List<Map> 将偏平化的数组转为树状结构并排序
数据格式: [ { "id":"d3e8a9d6-e4c6-4dd8-a94f-07733d3c1b59", "parentId":&quo ...
随机推荐
- OC数组和字典中存入niu值
在NSArray和NSDictionary中nil有特殊的含义.但是某些时候,我们必须要放入nil怎么办? 要想放入nil就必须用到一个类NSNull,这个类只有一个类方法,就是null.[NSNul ...
- BZOJ 4771 主席树+倍增+set
思路: 因为有深度的限制,并且我们是在线段树上维护权值,所以我们把点按照dep排序,然后一个一个修改...主席树的下标就是dfs序,子树的查询就是区间查询... 但是发现这样怎么去维护LCA呢...因 ...
- IAAS: IT公司去IOE-Alibaba系统构架解读
从Hadoop到自主研发,技术解读阿里去IOE后的系统架构 原地址:...................... 云计算阿里飞天 摘要:从IOE时代,到Hadoop与飞天并行,再到飞天单集群5000节 ...
- Swift语法3.03(类型Types)
类型 在Swift中,有两种类型:命名型类型和复合型类型.命名型类型是在定义时可以给定的特定名字的类型.命名型类型包括类,结构体,枚举和协议.例如,自定义的类MyClass的实例拥有类型MyClass ...
- SpringMVC(一) HelloWorld
学习新东西的的第一个程序--HelloWorld,以下是SpringMVC的HelloWorld 第一步: 用MAVEN 创建webapp,并添加依赖.(强烈建议使用MAVEN,MAVEN学习书籍和视 ...
- 遍历及过滤 first(), last() 和 eq() filter() 和 not()
三个最基本的过滤方法是:first(), last() 和 eq(),它们允许您基于其在一组元素中的位置来选择一个特定的元素.其他过滤方法,比如 filter() 和 not() 允许您选取匹配或不匹 ...
- 实现数组类(C++ 拷贝构造函数、拷贝函数)要判断赋值左右对象不相等,坑惨了
#include <iostream> using namespace std; class ArrayIndexOutOfBoundsException{ // 异常类 public: ...
- AOJ 2224 Save your cats( 最小生成树 )
链接:传送门 题意:有个女巫把猫全部抓走放在一个由 n 个木桩(xi,yi),m 个篱笆(起点终点木桩的编号)围成的法术领域内,我们必须用圣水才能将篱笆打开,然而圣水非常贵,所以我们尽量想降低花费来解 ...
- linux 编译网卡驱动
将smsc7500网卡驱动拷贝到/drive/net/usb文件夹下 拷贝ioctl_7500.h smsc7500usbnet.c smsc7500version.h smsclan7500.h ...
- Warning: The following processes: -cmd.exe -java.exe are locking the following directory: