bzoj 4811 由乃的OJ

  • 考虑树链剖分.
  • 树剖后用一颗线段树维护一段连续区间,类似于一个函数,各位上进入 \(0/1\) ,输出的数字分别是什么.注意到最多只有 \(64\) 位,可以用一个 \(unsigned\ long\ long\) 的大数状压表示,合并两段区间时推导一下可以做到 \(O(1)\) .
  • 注意 \(3 种\)位运算混在一起,满足交换律,却不满足结合律,所以从区间左/右侧进入同一个数,最后得到的数不同.需要维护两个方向的函数.
  • 修改时在线段树上单点修改就好,查询用树剖的基本操作,先找出 \(x\to y\) 这一段的函数,再贪心构造解.
#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define mp make_pair
#define pii pair<int,int>
typedef unsigned long long ull;
ull read(){
ull nm=0,fh=1;char cw=getchar();
for(;!isdigit(cw);cw=getchar()) ;
for(;isdigit(cw);cw=getchar()) nm=nm*10+(cw-'0');
return nm*fh;
}
const int MAXN=2e5+10;
ull cnt=0,head[MAXN],to[MAXN<<1],nx[MAXN<<1];
inline void addedge(ull u,ull v)
{
++cnt;
to[cnt]=v;
nx[cnt]=head[u];
head[u]=cnt;
}
inline void ins(ull u,ull v)
{
addedge(u,v);
addedge(v,u);
}
const unsigned long long U1=1;
ull fa[MAXN],siz[MAXN],mxson[MAXN],top[MAXN],dfn[MAXN],rnk[MAXN],dep[MAXN],idx=0;
void dfs1(ull u)
{
siz[u]=U1;
dep[u]=dep[fa[u]]+U1;
for(ull i=head[u];i>0;i=nx[i])
{
ull v=to[i];
if(v==fa[u])
continue;
fa[v]=u;
dfs1(v);
siz[u]+=siz[v];
if(siz[v]>siz[mxson[u]])
mxson[u]=v;
}
}
void dfs2(ull u,ull Tp)
{
dfn[u]=++idx;
rnk[idx]=u;
top[u]=Tp;
if(mxson[u]>0)
dfs2(mxson[u],Tp);
for(ull i=head[u];i>0;i=nx[i])
{
ull v=to[i];
if(v==fa[u] || v==mxson[u])
continue;
dfs2(v,v);
}
}
inline ull calc(ull x,ull bas,ull op)
{
if(op==1)
return x&bas;
if(op==2)
return x|bas;
return x^bas;
}
ull val[MAXN],opt[MAXN];
unsigned long long maxv=0;
struct node{
ull l,r;
ull d[2][2];//d[左入/右入][全0/全1]
void init(ull v,ull op)
{
d[0][0]=d[1][0]=calc(0,v,op);
d[0][1]=d[1][1]=calc(maxv,v,op);
}
void merge(node L,node R)
{
d[0][0]=(L.d[0][0]&R.d[0][1])|((~L.d[0][0])&R.d[0][0]);
d[1][0]=(R.d[1][0]&L.d[1][1])|((~R.d[1][0])&L.d[1][0]);
d[0][1]=(L.d[0][1]&R.d[0][1])|((~L.d[0][1])&R.d[0][0]);
d[1][1]=(R.d[1][1]&L.d[1][1])|((~R.d[1][1])&L.d[1][0]);
}
void inverse()
{
swap(d[0][0],d[1][0]);
swap(d[0][1],d[1][1]);
}
};
struct SegmentTree{
#define root Tree[o]
#define lson Tree[o<<1]
#define rson Tree[o<<1|1]
node Tree[MAXN<<2];
void BuildTree(ull o,ull l,ull r)
{
root.l=l,root.r=r;
if(l==r)
{
root.init(val[rnk[l]],opt[rnk[l]]);
return;
}
ull mid=(l+r)>>1;
BuildTree(o<<1,l,mid);
BuildTree(o<<1|1,mid+1,r);
root.merge(lson,rson);
}
void update(ull o,ull pos,ull newv,ull newopt)
{
ull l=root.l,r=root.r;
if(l==r)
{
root.init(newv,newopt);
return;
}
ull mid=(l+r)>>1;
if(pos<=mid)
update(o<<1,pos,newv,newopt);
else
update(o<<1|1,pos,newv,newopt);
root.merge(lson,rson);
}
node query(ull o,ull L,ull R)
{
ull l=root.l,r=root.r;
if(L<=l && r<=R)
return root;
node res;
res.init(0,3);
if(l>R || r<L)
return res;
res.merge(query(o<<1,L,R),query(o<<1|1,L,R));
return res;
}
}T;
ull n,m,k;
void init()
{
ull rt=(n+1)>>1;
fa[rt]=0;
for(ull i=0;i<k;i++)
maxv+=(U1<<i);
dfs1(rt);
dfs2(rt,rt);
T.BuildTree(1,1,n);
}
ull solve(ull x,ull y,ull lim)
{
ull tot=0;
node res1,res2,cur;
res1.init(0,3);
res2.init(0,3);
while(top[x]!=top[y])
{
if(dep[top[x]]>dep[top[y]])
{
res1.merge(T.query(1,dfn[top[x]],dfn[x]),res1);
x=fa[top[x]];
}
else
{
res2.merge(T.query(1,dfn[top[y]],dfn[y]),res2);
y=fa[top[y]];
}
}
if(dep[x]>dep[y])
{
node c=T.query(1,dfn[y],dfn[x]);
res1.merge(c,res1);
}
else
{
node c=T.query(1,dfn[x],dfn[y]);
res2.merge(c,res2);
}
res1.inverse();
cur.merge(res1,res2);
for(ull i=k;i-->0;)
{
if(cur.d[0][0] & (U1<<i))
tot+=(U1<<i);
else if( ( (cur.d[0][1]) & (U1<<i) ) >0 && lim>=(U1<<i) )
lim-=(U1<<i),tot+=(U1<<i);
}
return tot;
}
int main()
{
n=read(),m=read(),k=read();
for(ull i=1;i<=n;++i)
opt[i]=read(),val[i]=read();
for(ull i=1;i<n;++i)
{
ull u=read(),v=read();
ins(u,v);
}
init();
while(m--)
{
ull op=read();
ull x=read(),y=read(),z=read();
if(op==1)
cout<<solve(x,y,z)<<endl;
else
T.update(1,dfn[x],z,y);
}
return 0;
}

bzoj 4811 由乃的OJ的更多相关文章

  1. bzoj 4811: [Ynoi2017]由乃的OJ

    树链剖分,用zkw线段树维护每条链两个方向上对每一位的变换情况,由于位数较少,可以用两个unsigned long long表示 #include<cstdio> typedef unsi ...

  2. BZOJ 4811 [Ynoi2017]由乃的OJ ——Link-Cut Tree

    直接维护按照顺序经过每一段,初始的1可以变成什么,初始为0可以变成什么. 然后答案就可以和起床困难综合征一样贪心处理了. 写起来并不好写. 发现交换左右子树之后答案会改变,GG 调了一天,最后还是T掉 ...

  3. BZOJ 4811 树链剖分+线段树

    思路: 感觉这题也可神了.. (还是我太弱) 首先发现每一位不会互相影响,可以把每一位分开考虑,然后用树链剖分或者LCT维护这个树 修改直接修改,询问的时候算出来每一位填0,1经过这条链的变换之后得到 ...

  4. bzoj 1565 [NOI2009]植物大战僵尸 解题报告

    1565: [NOI2009]植物大战僵尸 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 2161  Solved: 1000[Submit][Stat ...

  5. 博主自传——蒟蒻的OI之路

    博主来自河北石家庄市第二中学,现在读高二,主攻信息学竞赛(其实并没有学习其他学科竞赛). NOIP中人品大爆发,使劲挤进河北省一等奖队伍,侥幸留在竞赛团队中(差点就淘汰出局啦). 关于我的ID,YOU ...

  6. 【BZOJ】3224: Tyvj 1728 普通平衡树(某不科学的oj)

    http://www.lydsy.com/JudgeOnline/problem.php?id=3224 无力吐槽,无力吐槽,无力吐槽....... bzoj竟然不能用time(0)我竟然不造!!re ...

  7. BZOJ 3595: [Scoi2014]方伯伯的Oj SBT+可持久化Treap

    3595: [Scoi2014]方伯伯的Oj Time Limit: 6 Sec  Memory Limit: 256 MBSubmit: 102  Solved: 54[Submit][Status ...

  8. bzoj 3678 wangxz与OJ

    3678: wangxz与OJ Time Limit: 10 Sec  Memory Limit: 128 MBhttp://www.lydsy.com/JudgeOnline/problem.php ...

  9. BZOJ 3595: [Scoi2014]方伯伯的Oj Splay + 动态裂点 + 卡常

    Description 方伯伯正在做他的Oj.现在他在处理Oj上的用户排名问题. Oj上注册了n个用户,编号为1-”,一开始他们按照编号排名.方伯伯会按照心情对这些用户做以下四种操作,修改用户的排名和 ...

随机推荐

  1. ABP 源码分析汇总之 AutoMapper

    AutoMapper 是一个对象映射工具, 安装时只需要安装 如下即可: 有关于它的介绍,参考官网:http://automapper.org/ AutoMapper使用比较简单,还是直奔主题,看一下 ...

  2. [转载]Eclipse的常用快捷键

    常用的快捷键 ctrl+1:快速修复错误 ctrl+shift+L :查看快捷键 alt+?或alt+/:自动补全代码或者提示代码 ctrl+o:快速outline视图 ctrl+shift+r:打开 ...

  3. MyEclipse2014.Maven自动更新

    1.我把 "Do not automatically update dependencies from remote repositories" 和 "Download ...

  4. [Vue]组件——通过$emit为组件自定义事件

    1.在定义组件时调用内建的 $emit 方法并传入事件的名字,来向父级组件触发一个事件enlarge-text: Vue.component('blog-post', { props: ['post' ...

  5. spring mvc: 参数方法名称解析器(用参数来解析控制器下的方法)MultiActionController/ParameterMethodNameResolver/ControllerClassNameHandlerMapping

    spring mvc: 参数方法名称解析器(用参数来解析控制器下的方法)MultiActionController/ParameterMethodNameResolver/ControllerClas ...

  6. HDU 4745 Two Rabbits ★(最长回文子序列:区间DP)

    题意 在一个圆环串中找一个最长的子序列,并且这个子序列是轴对称的. 思路 从对称轴上一点出发,向两个方向运动可以正好满足题意,并且可以证明如果抽选择的子环不是对称的话,其一定不是最长的. 倍长原序列, ...

  7. Highcharts 时间序列,可缩放的图表;Highcharts X 轴翻转曲线图;Highcharts 带标记曲线图

    Highcharts 时间序列,可缩放的图表 配置 图表 配置可缩放图表. chart.zoomType 指定了用户可以拖放的尺寸,用户可以通过拖动鼠标来放大,可能值是x,y或xy: var char ...

  8. 【Java实战】源码解析Java SPI(Service Provider Interface )机制原理

    一.背景知识 在阅读开源框架源码时,发现许多框架都支持SPI(Service Provider Interface ),前面有篇文章JDBC对Driver的加载时应用了SPI,参考[Hibernate ...

  9. android 几个开源项目

    android的几个开源项目ormlite.volley.jsoup.vitamio ksoap2

  10. 看完前任三,想起我的前任java女程序员,她曾教会我……

    前任三最近非常火了,票房蹭蹭往上升,昨天也去电影院看了,想起了我的前任,她是一名女程序员,为了让我学好java,她曾经亲自教我Java的算法,学算法是件非常重要的事,在这忍住回忆的悲伤,分享给你们. ...