日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

BZOJ3862Little Devil I——树链剖分+线段树

發布時間:2024/9/21 编程问答 31 豆豆
生活随笔 收集整理的這篇文章主要介紹了 BZOJ3862Little Devil I——树链剖分+线段树 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

題目大意:

給一棵樹,每條邊可能是黑色或白色(起始都是白色),有三種操作:

1、將u到v路徑上所有邊顏色翻轉(黑->白,白->黑)

2、將只有一個點在u到v路徑上的邊顏色翻轉

3、查詢u到v路徑上顏色為黑色的邊數

?

如果只有1、3操作很好做,直接樹鏈剖分+線段樹,重點是2操作(廢話)。

可以發現只有lca需要翻轉和它父節點之間的邊,因此將這條邊暴力翻轉。

現在剩下需要翻轉的就是u到v路徑上所有點與他們子節點之間的邊。

我們將u到v的路徑分成若干條重鏈,可以發現每條重鏈除了鏈低端的那個點,其他點要反轉的是他們與他們所有輕兒子之間的邊(這里注意特判lca少翻轉了一條輕邊)。

這么多邊顯然不能都翻轉,那么我們再開一棵線段樹,將重鏈上的點打標記,代表這些點與他們所有輕兒子之間的邊需要翻轉。

至于重鏈低端那個點還要暴力翻轉它與重兒子間的邊。

但現在只處理了每條重鏈,沒有考慮兩條鏈之間連接的那條邊?

假設y鏈接在x鏈下面,那么x鏈低端那個點需要翻轉的是除了一個輕邊之外的其他輕邊和一條重邊,那么只需要將重邊和不需要翻轉的輕邊翻轉之后再打標記就好了。

將邊下傳到點,開兩棵線段樹,一棵維護邊的顏色,另一棵維護給輕兒子翻轉的標記(這棵線段樹建議標記永久化),每次修改時注意鏈與鏈之間輕邊的翻轉即可。

#include<set> #include<map> #include<stack> #include<queue> #include<cmath> #include<cstdio> #include<vector> #include<bitset> #include<cstring> #include<iostream> #include<algorithm> #define ll long long using namespace std; int T; int n,m; int x,y; int tot; int opt; int num; int a[800010]; int b[800010]; int d[100010]; int f[100010]; int s[100010]; int to[200010]; int sum[800010]; int son[100010]; int top[100010]; int size[100010]; int head[100010]; int next[200010]; void add(int x,int y) {tot++;next[tot]=head[x];head[x]=tot;to[tot]=y; } void dfs(int x) {d[x]=d[f[x]]+1;size[x]=1;for(int i=head[x];i;i=next[i]){if(to[i]!=f[x]){f[to[i]]=x;dfs(to[i]);size[x]+=size[to[i]];if(size[to[i]]>size[son[x]]){son[x]=to[i];}}} } void dfs2(int x,int tp) {s[x]=++num;top[x]=tp;if(son[x]){dfs2(son[x],tp);}for(int i=head[x];i;i=next[i]){if(to[i]!=f[x]&&to[i]!=son[x]){dfs2(to[i],to[i]);}} } void updata(int rt,int l,int r,int L,int R) {if(L<=l&&r<=R){b[rt]^=1;return ;}int mid=(l+r)>>1;if(L<=mid){updata(rt<<1,l,mid,L,R);}if(R>mid){updata(rt<<1|1,mid+1,r,L,R);} } int downdata(int rt,int l,int r,int k) {if(l==r){return b[rt];}int res=b[rt];int mid=(l+r)>>1;if(k<=mid){return res^downdata(rt<<1,l,mid,k);}else{return res^downdata(rt<<1|1,mid+1,r,k);} } void pushup(int rt) {sum[rt]=sum[rt<<1]+sum[rt<<1|1]; } void pushdown(int rt,int l,int r) {if(a[rt]){int mid=(l+r)>>1;a[rt<<1]^=1;a[rt<<1|1]^=1;sum[rt<<1]=(mid-l+1)-sum[rt<<1];sum[rt<<1|1]=(r-mid)-sum[rt<<1|1];a[rt]=0;} } void change1(int rt,int l,int r,int k) {if(l==r){sum[rt]^=1;return ;}pushdown(rt,l,r);int mid=(l+r)>>1;if(k<=mid){change1(rt<<1,l,mid,k);}else{change1(rt<<1|1,mid+1,r,k);}pushup(rt); } void change2(int rt,int l,int r,int L,int R) {if(L<=l&&r<=R){a[rt]^=1;sum[rt]=(r-l+1)-sum[rt];return ;}pushdown(rt,l,r);int mid=(l+r)>>1;if(L<=mid){change2(rt<<1,l,mid,L,R);}if(R>mid){change2(rt<<1|1,mid+1,r,L,R);}pushup(rt); } int query1(int rt,int l,int r,int k) {if(l==r){return sum[rt];}pushdown(rt,l,r);int mid=(l+r)>>1;if(k<=mid){return query1(rt<<1,l,mid,k);}else{return query1(rt<<1|1,mid+1,r,k);} } int query2(int rt,int l,int r,int L,int R) {if(L<=l&&r<=R){return sum[rt];}pushdown(rt,l,r);int mid=(l+r)>>1;int res=0;if(L<=mid){res+=query2(rt<<1,l,mid,L,R);}if(R>mid){res+=query2(rt<<1|1,mid+1,r,L,R);}return res; } void rotate1(int x,int y) {while(top[x]!=top[y]){if(d[top[x]]<d[top[y]]){swap(x,y);}change2(1,1,n,s[top[x]],s[x]);x=f[top[x]];}if(d[x]>d[y]){swap(x,y);}if(x==y){return ;}change2(1,1,n,s[x]+1,s[y]); } void rotate2(int x,int y) {if(son[x]){change1(1,1,n,s[x]+1);}if(son[y]){change1(1,1,n,s[y]+1);}while(top[x]!=top[y]){if(d[top[x]]<d[top[y]]){swap(x,y);}updata(1,1,n,s[top[x]],s[x]);change1(1,1,n,s[top[x]]);x=f[top[x]];change1(1,1,n,s[x]+1);}if(d[x]>d[y]){swap(x,y);}change1(1,1,n,s[x]);change1(1,1,n,s[x]+1);updata(1,1,n,s[x],s[y]); } int ask(int x,int y) {int res=0;int ans;while(top[x]!=top[y]){if(d[top[x]]<d[top[y]]){swap(x,y);}if(top[x]!=x){res+=query2(1,1,n,s[top[x]]+1,s[x]);}ans=query1(1,1,n,s[top[x]]);x=f[top[x]];res+=(ans^downdata(1,1,n,s[x]));}if(d[x]>d[y]){swap(x,y);}if(x==y){return res;}res+=query2(1,1,n,s[x]+1,s[y]);return res; } int main() {scanf("%d",&T);while(T--){num=0;tot=0;memset(a,0,sizeof(a));memset(b,0,sizeof(b));memset(f,0,sizeof(f));memset(d,0,sizeof(d));memset(s,0,sizeof(s));memset(to,0,sizeof(to));memset(son,0,sizeof(son));memset(top,0,sizeof(top));memset(sum,0,sizeof(sum));memset(size,0,sizeof(size));memset(head,0,sizeof(head));memset(next,0,sizeof(next));scanf("%d",&n);for(int i=1;i<n;i++){scanf("%d%d",&x,&y);add(x,y);add(y,x);}dfs(1);dfs2(1,1);scanf("%d",&m);for(int i=1;i<=m;i++){scanf("%d%d%d",&opt,&x,&y);if(opt==1){rotate1(x,y);}else if(opt==2){rotate2(x,y);}else{printf("%d\n",ask(x,y));}}} }

轉載于:https://www.cnblogs.com/Khada-Jhin/p/9719205.html

總結

以上是生活随笔為你收集整理的BZOJ3862Little Devil I——树链剖分+线段树的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

主站蜘蛛池模板: 中文文字幕文字幕高清 | 久久精品—区二区三区舞蹈 | 精品人妻一区二 | 中文字幕手机在线视频 | 国产欧美日韩精品一区二区三区 | 精品交短篇合集 | 91综合精品 | 污污污www精品国产网站 | 青青操狠狠干 | 黄色大片av | 伊人夜夜| 欧美射图 | 青青草这里只有精品 | 黄网站免费在线 | 在线看www| 久久久亚洲精品无码 | www.白虎 | 91精品国自产 | 日本成人黄色 | 婷婷去俺也去 | 欧美黑人精品 | 亚洲GV成人无码久久精品 | 国产在线拍揄自揄拍无码视频 | 有码视频在线观看 | 丰满人妻一区二区三区53视频 | 日韩成人免费在线 | 亚洲一区二区三区观看 | 亚洲第一页在线 | 成人毛片一区二区三区 | 91高清在线免费观看 | 另类性姿势bbwbbw | 亚洲精品在线播放视频 | 国产伦精品一区二区三区免费视频 | 国产女主播一区 | 天天干精品 | 无码日韩人妻精品久久蜜桃 | 特级黄色一级片 | 中国久久 | 亚洲第一成网站 | 羞羞漫画在线 | 在线艹 | 国产视频一区在线播放 | gav在线| 久久综合伊人77777蜜臀 | 致单身男女免费观看完整版 | a一级免费视频 | 熟女肥臀白浆大屁股一区二区 | 中文字幕一区二区三区视频 | 国产色秀 | 久久久视 | 国产又粗又猛又黄又爽的视频 | 香蕉网址 | 在线视频一区二区 | 亚洲欧美一区二区三区情侣bbw | 快播91 | 麻豆传媒网站 | 日韩精品一区二区在线观看 | 狠狠躁18三区二区一区传媒剧情 | 国产乱码精品一区二区三区亚洲人 | 北条麻妃99精品青青久久 | 亚洲国产精品久久久久久6q | 波多野结衣高清电影 | 国产精品美女在线观看 | 一区二区三区欧美视频 | 国产欧美日韩三级 | 99久久久| 成人黄色激情网 | 国产精品国产av | 亚洲精品一区二三区 | 色七七桃花影院 | 999这里有精品 | 91成人短视频 | 日韩精品第二页 | 国产又粗又长又爽 | 欧美一区二区视频免费观看 | 国内自拍在线 | av在线不卡免费 | 久久av免费观看 | 九九热色 | 国产免费一区二区三区最新不卡 | 深夜视频在线观看 | 三区在线 | 欧美极品在线观看 | 看片久久 | 欧美大片在线观看 | 人妖一级片| 夜夜躁日日躁狠狠久久av | 国产一区二区在线电影 | 精品无码av一区二区三区不卡 | 三级网站在线 | 欧美黑人性受xxxx精品 | 亚洲欧美成人一区二区三区 | 78m78成人免费网站 | 亚洲欧美日韩在线不卡 | 综合婷婷 | 欧美日韩中文国产 | 日本www在线 | 爱爱小视频网站 | 中文字幕欧美人妻精品 |