[转载]完全版线段树 by notonlysuccess大牛
原文出处:http://www.notonlysuccess.com/
(好像现在这个博客已经挂掉了,在网上找到的全部都是转载)
今天在清北学堂听课,听到了一些很令人吃惊的消息。至于这消息具体是啥,等我晚上再说23333
我大概的看了一下所讲内容及实例,果不愧是大牛级别的人,讲的很详细,所以决定转载过来。
对原文作者致以崇高的敬意。
以下是转载内容。
【完全版】线段树
很早前写的那篇线段树专辑至今一直是本博客阅读点击量最大的一片文章,当时觉得挺自豪的,还去pku打广告,但是现在我自己都不太好意思去看那篇文章了,觉得当时的代码风格实在是太丑了,很多线段树的初学者可能就是看着这篇文章来练习的,如果不小心被我培养出了这么糟糕的风格,实在是过意不去,正好过几天又要给集训队讲解线段树,所以决定把这些题目重新写一遍,顺便把近年我接触到的一些新题更新上去~;并且学习了splay等更高级的数据结构后对线段树的体会有更深了一层,线段树的写法也就比以前飘逸,简洁且方便多了.
在代码前先介绍一些我的线段树风格:
- maxn是题目给的最大区间,而节点数要开4倍,确切的来说节点数要开大于maxn的最小2x的两倍
- lson和rson分辨表示结点的左儿子和右儿子,由于每次传参数的时候都固定是这几个变量,所以可以用预定于比较方便的表示
- 以前的写法是另外开两个个数组记录每个结点所表示的区间,其实这个区间不必保存,一边算一边传下去就行,只需要写函数的时候多两个参数,结合lson和rson的预定义可以很方便
- PushUP(int rt)是把当前结点的信息更新到父结点
- PushDown(int rt)是把当前结点的信息更新给儿子结点
- rt表示当前子树的根(root),也就是当前所在的结点
整理这些题目后我觉得线段树的题目整体上可以分成以下四个部分:
单点更新:最最基础的线段树,只更新叶子节点,然后把信息用PushUP(int r)这个函数更新上来
hdu1166 敌兵布阵
题意:O(-1)
思路:O(-1)
线段树功能:update:单点增减 query:区间求和
#include <cstdio> #define lson l , m , rt << 1 #define rson m + 1 , r , rt << 1 | 1 const int maxn = ; int sum[maxn<<]; void PushUP(int rt) { sum[rt] = sum[rt<<] + sum[rt<<|]; } void build(int l,int r,int rt) { if (l == r) { scanf("%d",&sum[rt]); return ; } int m = (l + r) >> ; build(lson); build(rson); PushUP(rt); } void update(int p,int add,int l,int r,int rt) { if (l == r) { sum[rt] += add; return ; } int m = (l + r) >> ; if (p <= m) update(p , add , lson); else update(p , add , rson); PushUP(rt); } int query(int L,int R,int l,int r,int rt) { if (L <= l && r <= R) { return sum[rt]; } int m = (l + r) >> ; int ret = ; if (L <= m) ret += query(L , R , lson); if (R > m) ret += query(L , R , rson); return ret; } int main() { int T , n; scanf("%d",&T); for (int cas = ; cas <= T ; cas ++) { printf("Case %d:\n",cas); scanf("%d",&n); build( , n , ); char op[]; while (scanf("%s",op)) { if (op[] == 'E') break; int a , b; scanf("%d%d",&a,&b); if (op[] == 'Q') printf("%d\n",query(a , b , , n , )); else if (op[] == 'S') update(a , -b , , n , ); else update(a , b , , n , ); } } return ; }
hdu1754 I Hate It
题意:O(-1)
思路:O(-1)
线段树功能:update:单点替换 query:区间最值
#include <cstdio> #include <algorithm> using namespace std; #define lson l , m , rt << 1 #define rson m + 1 , r , rt << 1 | 1 const int maxn = ; int MAX[maxn<<]; void PushUP(int rt) { MAX[rt] = max(MAX[rt<<] , MAX[rt<<|]); } void build(int l,int r,int rt) { if (l == r) { scanf("%d",&MAX[rt]); return ; } int m = (l + r) >> ; build(lson); build(rson); PushUP(rt); } void update(int p,int sc,int l,int r,int rt) { if (l == r) { MAX[rt] = sc; return ; } int m = (l + r) >> ; if (p <= m) update(p , sc , lson); else update(p , sc , rson); PushUP(rt); } int query(int L,int R,int l,int r,int rt) { if (L <= l && r <= R) { return MAX[rt]; } int m = (l + r) >> ; int ret = ; if (L <= m) ret = max(ret , query(L , R , lson)); if (R > m) ret = max(ret , query(L , R , rson)); return ret; } int main() { int n , m; while (~scanf("%d%d",&n,&m)) { build( , n , ); while (m --) { char op[]; int a , b; scanf("%s%d%d",op,&a,&b); if (op[] == 'Q') printf("%d\n",query(a , b , , n , )); else update(a , b , , n , ); } } return ; }
hdu1394 Minimum Inversion Number
题意:求Inversion后的最小逆序数
思路:用O(nlogn)复杂度求出最初逆序数后,就可以用O(1)的复杂度分别递推出其他解
线段树功能:update:单点增减 query:区间求和
#include <cstdio> #include <algorithm> using namespace std; #define lson l , m , rt << 1 #define rson m + 1 , r , rt << 1 | 1 const int maxn = ; int sum[maxn<<]; void PushUP(int rt) { sum[rt] = sum[rt<<] + sum[rt<<|]; } void build(int l,int r,int rt) { sum[rt] = ; if (l == r) return ; int m = (l + r) >> ; build(lson); build(rson); } void update(int p,int l,int r,int rt) { if (l == r) { sum[rt] ++; return ; } int m = (l + r) >> ; if (p <= m) update(p , lson); else update(p , rson); PushUP(rt); } int query(int L,int R,int l,int r,int rt) { if (L <= l && r <= R) { return sum[rt]; } int m = (l + r) >> ; int ret = ; if (L <= m) ret += query(L , R , lson); if (R > m) ret += query(L , R , rson); return ret; } int x[maxn]; int main() { int n; while (~scanf("%d",&n)) { build( , n - , ); int sum = ; for (int i = ; i < n ; i ++) { scanf("%d",&x[i]); sum += query(x[i] , n - , , n - , ); update(x[i] , , n - , ); } int ret = sum; for (int i = ; i < n ; i ++) { sum += n - x[i] - x[i] - ; ret = min(ret , sum); } printf("%d\n",ret); } return ; }
hdu2795 Billboard
题意:h*w的木板,放进一些1*L的物品,求每次放空间能容纳且最上边的位子
思路:每次找到最大值的位子,然后减去L
线段树功能:query:区间求最大值的位子(直接把update的操作在query里做了)
#include <cstdio> #include <algorithm> using namespace std; #define lson l , m , rt << 1 #define rson m + 1 , r , rt << 1 | 1 const int maxn = ; int h , w , n; int MAX[maxn<<]; void PushUP(int rt) { MAX[rt] = max(MAX[rt<<] , MAX[rt<<|]); } void build(int l,int r,int rt) { MAX[rt] = w; if (l == r) return ; int m = (l + r) >> ; build(lson); build(rson); } int query(int x,int l,int r,int rt) { if (l == r) { MAX[rt] -= x; return l; } int m = (l + r) >> ; int ret = (MAX[rt<<] >= x) ? query(x , lson) : query(x , rson); PushUP(rt); return ret; } int main() { while (~scanf("%d%d%d",&h,&w,&n)) { if (h > n) h = n; build( , h , ); while (n --) { int x; scanf("%d",&x); if (MAX[] < x) puts("-1"); else printf("%d\n",query(x , , h , )); } } return ; }
练习:
poj2828 Buy Tickets
poj2886 Who Gets the Most Candies?
成段更新(通常这对初学者来说是一道坎),需要用到延迟标记(或者说懒惰标记),简单来说就是每次更新的时候不要更新到底,用延迟标记使得更新延迟到下次需要更新or询问到的时候
hdu1698 Just a Hook
题意:O(-1)
思路:O(-1)
线段树功能:update:成段替换 (由于只query一次总区间,所以可以直接输出1结点的信息)
#include <cstdio> #include <algorithm> using namespace std; #define lson l , m , rt << 1 #define rson m + 1 , r , rt << 1 | 1 const int maxn = ; int h , w , n; int col[maxn<<]; int sum[maxn<<]; void PushUp(int rt) { sum[rt] = sum[rt<<] + sum[rt<<|]; } void PushDown(int rt,int m) { if (col[rt]) { col[rt<<] = col[rt<<|] = col[rt]; sum[rt<<] = (m - (m >> )) * col[rt]; sum[rt<<|] = (m >> ) * col[rt]; col[rt] = ; } } void build(int l,int r,int rt) { col[rt] = ; sum[rt] = ; if (l == r) return ; int m = (l + r) >> ; build(lson); build(rson); PushUp(rt); } void update(int L,int R,int c,int l,int r,int rt) { if (L <= l && r <= R) { col[rt] = c; sum[rt] = c * (r - l + ); return ; } PushDown(rt , r - l + ); int m = (l + r) >> ; if (L <= m) update(L , R , c , lson); if (R > m) update(L , R , c , rson); PushUp(rt); } int main() { int T , n , m; scanf("%d",&T); for (int cas = ; cas <= T ; cas ++) { scanf("%d%d",&n,&m); build( , n , ); while (m --) { int a , b , c; scanf("%d%d%d",&a,&b,&c); update(a , b , c , , n , ); } printf("Case %d: The total value of the hook is %d.\n",cas , sum[]); } return ; }
poj3468 A Simple Problem with Integers
题意:O(-1)
思路:O(-1)
线段树功能:update:成段增减 query:区间求和
#include <cstdio> #include <algorithm> using namespace std; #define lson l , m , rt << 1 #define rson m + 1 , r , rt << 1 | 1 #define LL long long const int maxn = ; LL add[maxn<<]; LL sum[maxn<<]; void PushUp(int rt) { sum[rt] = sum[rt<<] + sum[rt<<|]; } void PushDown(int rt,int m) { if (add[rt]) { add[rt<<] += add[rt]; add[rt<<|] += add[rt]; sum[rt<<] += add[rt] * (m - (m >> )); sum[rt<<|] += add[rt] * (m >> ); add[rt] = ; } } void build(int l,int r,int rt) { add[rt] = ; if (l == r) { scanf("%lld",&sum[rt]); return ; } int m = (l + r) >> ; build(lson); build(rson); PushUp(rt); } void update(int L,int R,int c,int l,int r,int rt) { if (L <= l && r <= R) { add[rt] += c; sum[rt] += (LL)c * (r - l + ); return ; } PushDown(rt , r - l + ); int m = (l + r) >> ; if (L <= m) update(L , R , c , lson); if (m < R) update(L , R , c , rson); PushUp(rt); } LL query(int L,int R,int l,int r,int rt) { if (L <= l && r <= R) { return sum[rt]; } PushDown(rt , r - l + ); int m = (l + r) >> ; LL ret = ; if (L <= m) ret += query(L , R , lson); if (m < R) ret += query(L , R , rson); return ret; } int main() { int N , Q; scanf("%d%d",&N,&Q); build( , N , ); while (Q --) { char op[]; int a , b , c; scanf("%s",op); if (op[] == 'Q') { scanf("%d%d",&a,&b); printf("%lld\n",query(a , b , , N , )); } else { scanf("%d%d%d",&a,&b,&c); update(a , b , c , , N , ); } } return ; }
poj2528 Mayor’s posters
题意:在墙上贴海报,海报可以互相覆盖,问最后可以看见几张海报
思路:这题数据范围很大,直接搞超时+超内存,需要离散化:
离散化简单的来说就是只取我们需要的值来用,比如说区间[1000,2000],[1990,2012] 我们用不到[-∞,999][1001,1989][1991,1999][2001,2011][2013,+∞]这些值,所以我只需要1000,1990,2000,2012就够了,将其分别映射到0,1,2,3,在于复杂度就大大的降下来了
所以离散化要保存所有需要用到的值,排序后,分别映射到1~n,这样复杂度就会小很多很多
而这题的难点在于每个数字其实表示的是一个单位长度(并且一个点),这样普通的离散化会造成许多错误(包括我以前的代码,poj这题数据奇弱)
给出下面两个简单的例子应该能体现普通离散化的缺陷:
1-10 1-4 5-10
1-10 1-4 6-10
为了解决这种缺陷,我们可以在排序后的数组上加些处理,比如说[1,2,6,10]
如果相邻数字间距大于1的话,在其中加上任意一个数字,比如加成[1,2,3,6,7,10],然后再做线段树就好了.
线段树功能:update:成段替换 query:简单hash
#include <cstdio> #include <cstring> #include <algorithm> using namespace std; #define lson l , m , rt << 1 #define rson m + 1 , r , rt << 1 | 1 const int maxn = ; bool hash[maxn]; int li[maxn] , ri[maxn]; int X[maxn*]; int col[maxn<<]; int cnt; void PushDown(int rt) { if (col[rt] != -) { col[rt<<] = col[rt<<|] = col[rt]; col[rt] = -; } } void update(int L,int R,int c,int l,int r,int rt) { if (L <= l && r <= R) { col[rt] = c; return ; } PushDown(rt); int m = (l + r) >> ; if (L <= m) update(L , R , c , lson); if (m < R) update(L , R , c , rson); } void query(int l,int r,int rt) { if (col[rt] != -) { if (!hash[col[rt]]) cnt ++; hash[ col[rt] ] = true; return ; } if (l == r) return ; int m = (l + r) >> ; query(lson); query(rson); } int Bin(int key,int n,int X[]) { int l = , r = n - ; while (l <= r) { int m = (l + r) >> ; if (X[m] == key) return m; if (X[m] < key) l = m + ; else r = m - ; } return -; } int main() { int T , n; scanf("%d",&T); while (T --) { scanf("%d",&n); int nn = ; for (int i = ; i < n ; i ++) { scanf("%d%d",&li[i] , &ri[i]); X[nn++] = li[i]; X[nn++] = ri[i]; } sort(X , X + nn); int m = ; for (int i = ; i < nn; i ++) { if (X[i] != X[i-]) X[m ++] = X[i]; } for (int i = m - ; i > ; i --) { if (X[i] != X[i-] + ) X[m ++] = X[i] + ; } sort(X , X + m); memset(col , - , sizeof(col)); for (int i = ; i < n ; i ++) { int l = Bin(li[i] , m , X); int r = Bin(ri[i] , m , X); update(l , r , i , , m , ); } cnt = ; memset(hash , false , sizeof(hash)); query( , m , ); printf("%d\n",cnt); } return ; }
poj3225 Help with Intervals
题意:区间操作,交,并,补等
思路:
我们一个一个操作来分析:(用0和1表示是否包含区间,-1表示该区间内既有包含又有不包含)
U:把区间[l,r]覆盖成1
I:把[-∞,l)(r,∞]覆盖成0
D:把区间[l,r]覆盖成0
C:把[-∞,l)(r,∞]覆盖成0 , 且[l,r]区间0/1互换
S:[l,r]区间0/1互换
成段覆盖的操作很简单,比较特殊的就是区间0/1互换这个操作,我们可以称之为异或操作
很明显我们可以知道这个性质:当一个区间被覆盖后,不管之前有没有异或标记都没有意义了
所以当一个节点得到覆盖标记时把异或标记清空
而当一个节点得到异或标记的时候,先判断覆盖标记,如果是0或1,直接改变一下覆盖标记,不然的话改变异或标记
开区间闭区间只要数字乘以2就可以处理(偶数表示端点,奇数表示两端点间的区间)
线段树功能:update:成段替换,区间异或 query:简单hash
#include <cstdio> #include <cstring> #include <cctype> #include <algorithm> using namespace std; #define lson l , m , rt << 1 #define rson m + 1 , r , rt << 1 | 1 const int maxn = ; bool hash[maxn]; int cover[maxn<<]; int XOR[maxn<<]; void FXOR(int rt) { if (cover[rt] != -) cover[rt] ^= ; else XOR[rt] ^= ; } void PushDown(int rt) { if (cover[rt] != -) { cover[rt<<] = cover[rt<<|] = cover[rt]; XOR[rt<<] = XOR[rt<<|] = ; cover[rt] = -; } if (XOR[rt]) { FXOR(rt<<); FXOR(rt<<|); XOR[rt] = ; } } void update(char op,int L,int R,int l,int r,int rt) { if (L <= l && r <= R) { if (op == 'U') { cover[rt] = ; XOR[rt] = ; } else if (op == 'D') { cover[rt] = ; XOR[rt] = ; } else if (op == 'C' || op == 'S') { FXOR(rt); } return ; } PushDown(rt); int m = (l + r) >> ; if (L <= m) update(op , L , R , lson); else if (op == 'I' || op == 'C') { XOR[rt<<] = cover[rt<<] = ; } if (m < R) update(op , L , R , rson); else if (op == 'I' || op == 'C') { XOR[rt<<|] = cover[rt<<|] = ; } } void query(int l,int r,int rt) { if (cover[rt] == ) { for (int it = l ; it <= r ; it ++) { hash[it] = true; } return ; } else if (cover[rt] == ) return ; if (l == r) return ; PushDown(rt); int m = (l + r) >> ; query(lson); query(rson); } int main() { cover[] = XOR[] = ; char op , l , r; int a , b; while ( ~scanf("%c %c%d,%d%c\n",&op , &l , &a , &b , &r) ) { a <<= , b <<= ; if (l == '(') a ++; if (r == ')') b --; if (a > b) { if (op == 'C' || op == 'I') { cover[] = XOR[] = ; } } else update(op , a , b , , maxn , ); } query( , maxn , ); bool flag = false; int s = - , e; for (int i = ; i <= maxn ; i ++) { if (hash[i]) { if (s == -) s = i; e = i; } else { if (s != -) { if (flag) printf(" "); flag = true; printf("%c%d,%d%c",s&?'(':'[' , s>> , (e+)>> , e&?')':']'); s = -; } } } if (!flag) printf("empty set"); puts(""); return ; }
练习:
poj1436 Horizontally Visible Segments
poj2991 Crane
Another LCIS
Bracket Sequence
区间合并
这类题目会询问区间中满足条件的连续最长区间,所以PushUp的时候需要对左右儿子的区间进行合并
poj3667 Hotel
题意:1 a:询问是不是有连续长度为a的空房间,有的话住进最左边
2 a b:将[a,a+b-1]的房间清空
思路:记录区间中最长的空房间
线段树操作:update:区间替换 query:询问满足条件的最左断点
#include <cstdio> #include <cstring> #include <cctype> #include <algorithm> using namespace std; #define lson l , m , rt << 1 #define rson m + 1 , r , rt << 1 | 1 const int maxn = ; int lsum[maxn<<] , rsum[maxn<<] , msum[maxn<<]; int cover[maxn<<]; void PushDown(int rt,int m) { if (cover[rt] != -) { cover[rt<<] = cover[rt<<|] = cover[rt]; msum[rt<<] = lsum[rt<<] = rsum[rt<<] = cover[rt] ? : m - (m >> ); msum[rt<<|] = lsum[rt<<|] = rsum[rt<<|] = cover[rt] ? : (m >> ); cover[rt] = -; } } void PushUp(int rt,int m) { lsum[rt] = lsum[rt<<]; rsum[rt] = rsum[rt<<|]; if (lsum[rt] == m - (m >> )) lsum[rt] += lsum[rt<<|]; if (rsum[rt] == (m >> )) rsum[rt] += rsum[rt<<]; msum[rt] = max(lsum[rt<<|] + rsum[rt<<] , max(msum[rt<<] , msum[rt<<|])); } void build(int l,int r,int rt) { msum[rt] = lsum[rt] = rsum[rt] = r - l + ; cover[rt] = -; if (l == r) return ; int m = (l + r) >> ; build(lson); build(rson); } void update(int L,int R,int c,int l,int r,int rt) { if (L <= l && r <= R) { msum[rt] = lsum[rt] = rsum[rt] = c ? : r - l + ; cover[rt] = c; return ; } PushDown(rt , r - l + ); int m = (l + r) >> ; if (L <= m) update(L , R , c , lson); if (m < R) update(L , R , c , rson); PushUp(rt , r - l + ); } int query(int w,int l,int r,int rt) { if (l == r) return l; PushDown(rt , r - l + ); int m = (l + r) >> ; if (msum[rt<<] >= w) return query(w , lson); else if (rsum[rt<<] + lsum[rt<<|] >= w) return m - rsum[rt<<] + ; return query(w , rson); } int main() { int n , m; scanf("%d%d",&n,&m); build( , n , ); while (m --) { int op , a , b; scanf("%d",&op); if (op == ) { scanf("%d",&a); if (msum[] < a) puts(""); else { int p = query(a , , n , ); printf("%d\n",p); update(p , p + a - , , , n , ); } } else { scanf("%d%d",&a,&b); update(a , a + b - , , , n , ); } } return ; }
练习:
hdu3308 LCIS
hdu3397 Sequence operation
hdu2871 Memory Control
hdu1540 Tunnel Warfare
CF46-D Parking Lot
扫描线
这类题目需要将一些操作排序,然后从左到右用一根扫描线(当然是在我们脑子里)扫过去
最典型的就是矩形面积并,周长并等题
hdu1542 Atlantis
题意:矩形面积并
思路:浮点数先要离散化;然后把矩形分成两条边,上边和下边,对横轴建树,然后从下到上扫描上去,用cnt表示该区间下边比上边多几个
线段树操作:update:区间增减 query:直接取根节点的值
#include <cstdio> #include <cstring> #include <cctype> #include <algorithm> using namespace std; #define lson l , m , rt << 1 #define rson m + 1 , r , rt << 1 | 1 const int maxn = ; int cnt[maxn << ]; double sum[maxn << ]; double X[maxn]; struct Seg { double h , l , r; int s; Seg(){} Seg(double a,double b,double c,int d) : l(a) , r(b) , h(c) , s(d) {} bool operator < (const Seg &cmp) const { return h < cmp.h; } }ss[maxn]; void PushUp(int rt,int l,int r) { if (cnt[rt]) sum[rt] = X[r+] - X[l]; else if (l == r) sum[rt] = ; else sum[rt] = sum[rt<<] + sum[rt<<|]; } void update(int L,int R,int c,int l,int r,int rt) { if (L <= l && r <= R) { cnt[rt] += c; PushUp(rt , l , r); return ; } int m = (l + r) >> ; if (L <= m) update(L , R , c , lson); if (m < R) update(L , R , c , rson); PushUp(rt , l , r); } int Bin(double key,int n,double X[]) { int l = , r = n - ; while (l <= r) { int m = (l + r) >> ; if (X[m] == key) return m; if (X[m] < key) l = m + ; else r = m - ; } return -; } int main() { int n , cas = ; while (~scanf("%d",&n) && n) { int m = ; while (n --) { double a , b , c , d; scanf("%lf%lf%lf%lf",&a,&b,&c,&d); X[m] = a; ss[m++] = Seg(a , c , b , ); X[m] = c; ss[m++] = Seg(a , c , d , -); } sort(X , X + m); sort(ss , ss + m); int k = ; for (int i = ; i < m ; i ++) { if (X[i] != X[i-]) X[k++] = X[i]; } memset(cnt , , sizeof(cnt)); memset(sum , , sizeof(sum)); double ret = ; for (int i = ; i < m - ; i ++) { int l = Bin(ss[i].l , k , X); int r = Bin(ss[i].r , k , X) - ; if (l <= r) update(l , r , ss[i].s , , k - , ); ret += sum[] * (ss[i+].h - ss[i].h); } printf("Test case #%d\nTotal explored area: %.2lf\n\n",cas++ , ret); } return ; }
hdu1828 Picture
题意:矩形周长并
思路:与面积不同的地方是还要记录竖的边有几个(numseg记录),并且当边界重合的时候需要合并(用lbd和rbd表示边界来辅助)
线段树操作:update:区间增减 query:直接取根节点的值
#include <cstdio> #include <cstring> #include <cctype> #include <algorithm> using namespace std; #define lson l , m , rt << 1 #define rson m + 1 , r , rt << 1 | 1 const int maxn = ; struct Seg{ int l , r , h , s; Seg() {} Seg(int a,int b,int c,int d):l(a) , r(b) , h(c) , s(d) {} bool operator < (const Seg &cmp) const { return h < cmp.h; } }ss[maxn]; bool lbd[maxn<<] , rbd[maxn<<]; int numseg[maxn<<]; int cnt[maxn<<]; int len[maxn<<]; void PushUP(int rt,int l,int r) { if (cnt[rt]) { lbd[rt] = rbd[rt] = ; len[rt] = r - l + ; numseg[rt] = ; } else if (l == r) { len[rt] = numseg[rt] = lbd[rt] = rbd[rt] = ; } else { lbd[rt] = lbd[rt<<]; rbd[rt] = rbd[rt<<|]; len[rt] = len[rt<<] + len[rt<<|]; numseg[rt] = numseg[rt<<] + numseg[rt<<|]; if (lbd[rt<<|] && rbd[rt<<]) numseg[rt] -= ;//两条线重合 } } void update(int L,int R,int c,int l,int r,int rt) { if (L <= l && r <= R) { cnt[rt] += c; PushUP(rt , l , r); return ; } int m = (l + r) >> ; if (L <= m) update(L , R , c , lson); if (m < R) update(L , R , c , rson); PushUP(rt , l , r); } int main() { int n; while (~scanf("%d",&n)) { int m = ; int lbd = , rbd = -; for (int i = ; i < n ; i ++) { int a , b , c , d; scanf("%d%d%d%d",&a,&b,&c,&d); lbd = min(lbd , a); rbd = max(rbd , c); ss[m++] = Seg(a , c , b , ); ss[m++] = Seg(a , c , d , -); } sort(ss , ss + m); int ret = , last = ; for (int i = ; i < m ; i ++) { if (ss[i].l < ss[i].r) update(ss[i].l , ss[i].r - , ss[i].s , lbd , rbd - , ); ret += numseg[] * (ss[i+].h - ss[i].h); ret += abs(len[] - last); last = len[]; } printf("%d\n",ret); } return ; }
练习
hdu3265 Posters
hdu3642 Get The Treasury
poj2482 Stars in Your Window
poj2464 Brownie Points II
hdu3255 Farming
ural1707 Hypnotoad’s Secret
uva11983 Weird Advertisement
线段树与其他结合练习(欢迎大家补充):
hdu3333 Turing Tree
hdu3874 Necklace
hdu3016 Man Down
hdu3340 Rain in ACStar
zju3511 Cake Robbery
UESTC1558 Charitable Exchange
CF85-D Sum of Medians
spojGSS2 Can you answer these queries II
[转载]完全版线段树 by notonlysuccess大牛的更多相关文章
- 【转载】完全版线段树 by notonlysuccess大牛
原文出处:http://www.notonlysuccess.com/ 今晚上比赛就考到了 排兵布阵啊,难受. [完全版]线段树 很早前写的那篇线段树专辑至今一直是本博客阅读点击量最大的一片文章,当时 ...
- 《完全版线段树》——notonlysuccess
转载自:NotOnlySuccess的博客 [完全版]线段树 很早前写的那篇线段树专辑至今一直是本博客阅读点击量最大的一片文章,当时觉得挺自豪的,还去pku打广告,但是现在我自己都不太好意思去看那篇文 ...
- HDU 1166.敌兵布阵-完全版线段树(单点增减、区间求和)
生活艰辛,且行且珍惜. 先水一篇博客再去补题,要不然又忘记写博客了. 计划系统的刷一遍线段树专题,自己给自己找虐(自作孽不可活),从基础的到后面的,所有的都挂了题,刷题不,大兄弟? 线段树可真有意思, ...
- HDU 1754.I Hate It-结构体版线段树(单点更新+区间查询最值)
I Hate It Time Limit: 9000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total S ...
- POJ 3264.Balanced Lineup-结构体版线段树(区间查询最值)
Balanced Lineup Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 53721 Accepted: 25244 ...
- POJ 2828.Buy Tickets-完全版线段树(单点更新、逆序遍历查询)
POJ2828.Buy Tickets 这个题是插队问题,每次有人插队的时候,其后的所有数据都要进行更新,如果我们反着推,就可以把所有的数据都安排好并且不用再对已插入的数据进行更新,因为逆序处理的话所 ...
- HDU 2795.Billboard-完全版线段树(区间求最值的位置、区间染色、贴海报)
HDU2795.Billboard 这个题的意思就是在一块h*w的板子上贴公告,公告的规格为1*wi ,张贴的时候尽量往上,同一高度尽量靠左,求第n个公告贴的位置所在的行数,如果没有合适的位置贴则输出 ...
- HDU 1394.Minimum Inversion Number-最小逆序数-完全版线段树(单点增减、区间求和)
HDU1394.Minimum Inversion Number 这个题求最小逆序数,先建一个空的树,然后每输入一个值,就先查询一下,查询之后,更新线段树,然后遍历一遍,每次将第一个数放到最后之后,减 ...
- HDU 1754.I Hate It-完全版线段树(单点替换、区间最值)
HDU1754.I Hate It 直接模板就可以了 代码: //B #include<iostream> #include<cstdio> #include<cstri ...
随机推荐
- [国嵌攻略][067][tftp协议分析]
TFTP作用 用于网络下载,TFTP客户机在TFTP服务器中下载文件. TFTP交换过程 1.配置TFTP服务器 vim /etc/xinetd.d/tftp 2.交换过程 客户端发请求包到服务器 服 ...
- POJ 2387 Til the Cows Come Home(模板——Dijkstra算法)
题目连接: http://poj.org/problem?id=2387 Description Bessie is out in the field and wants to get back to ...
- DEDECMS首页调用图片集里的多张图片
本文给大家分享的是织梦系统中首页调用图片集里的多张图片的方法,有相同需要的小伙伴可以参考下. 先找到include/common.inc.php文件,把下面代码贴进去(我贴的是我网站上的,具体可根据需 ...
- dedecms_
2012-7-5(no1)当我们点击检索结果的某个电影超链接时,如何跳转到对应的内容页[本资源由www.qinglongweb.com搜集整理] dedelist标签 --可以嵌套 项目移植: mys ...
- vue中引入jQuery和bootstrap
一.引入jQuery: 首先在当前项目的根目录下(就是与package.json同目录),运行命令npm install jquery --save-dev 这样就将jquery安装到了这个项目中 ...
- mybatis_SQL缓存(5)
<settings> <!-- 这个配置使全局的映射器启用或禁用缓存 --> <setting name="cacheEnabled" value=& ...
- 导入Mybatis_Spring项目遇到的问题
1. 问题: jdk版本不匹配 解决方法:首先 到项目空间的 .setting文件中找到 org.eclipse.wst.common.project.facet.core.xml 修改参 ...
- sqllite小型数据库的使用
1.适用场景:免安装型数据库:数据量不大,本地化管理:不依赖其他第三方类库:2.具体使用方法:添加sqllite类库引用 数据库连接定义,数据库以文件形式存储在sqllitedb/solution.d ...
- 微信小程序左右滑动切换图片酷炫效果
开门见山,先上效果吧!感觉可以的用的上的再往下看. 心动吗?那就继续往下看! 先上页面结构吧,也就是wxml文件,其实可以理解成微信自己封装过的html,这个不多说了,不懂也没必要往下看了. < ...
- linux ngix安装
因为我完全按照第一篇参考文章从上到下一步步安装导致有些安装失败最后重装的,过程有点乱,就没自己总结please read the follow articles Linux 安装Nginx详细图解教程 ...