微信扫一扫打赏支持

区间覆盖(线段树)

区间覆盖(线段树)

X轴上方有若干条平行于X轴的线段,求这些线段能够覆盖X轴的总长度?
输入格式
第一行一个数n(n<=100000),表示线段个数;
接下来n行,每行两个整数a[i],b[i](-10^8<=a[i],b[i]<=10^8),代表一个线段的两个端点输出覆盖X轴的长度

输入样例
2
10 12
2 4
输出样例
4

 

 

 

 1 /*
 2 样例:
 3 有两段线
 4 1 2
 5 2 3 
 6 */
 7 
 8 #include <bits/stdc++.h>
 9 using namespace std;
10 void print();
11 struct node{
12     int start,end;
13     int cover;
14 }segTree[100];
15 
16 //创建线段树 
17 void build(int root,int start,int end){
18     segTree[root].start=start;
19     segTree[root].end=end;
20     segTree[root].cover=0;
21     if(start+1<end){
22         int mid=(start+end)/2;
23         build(2*root,start,mid);
24         build(2*root+1,mid,end);
25     }
26 }
27 
28 //1-4怎么办 
29 //插入线段,线段的左右端点分别为left和right 
30 void insert(int root,int left,int right){
31     //这里还是空的才需要插入 
32     if(segTree[root].cover==0){
33         int mid=(segTree[root].start+segTree[root].end)/2;
34         if(left==segTree[root].start&&right==segTree[root].end){
35             segTree[root].cover=1;
36         }
37         else if(right<mid){//去左子树中找 
38             insert(2*root,left,right);
39         }
40         else if(left>mid){//去右子树中找 
41             insert(2*root+1,left,right);
42         }
43         else{//如果没有正好合适的,就把区间给拆了 
44             insert(2*root,left,mid);
45             insert(2*root+1,mid,right);
46         } 
47     }
48 } 
49 
50 
51 int count(int root){
52     if(segTree[root].cover==1){
53         return segTree[root].end-segTree[root].start;
54     }
55     //又不等于1,又是叶子节点,肯定给你退出 
56     else if(segTree[root].end-segTree[root].start==1){
57         return 0;
58     }
59     else{
60         return count(2*root)+count(2*root+1); 
61     }
62 }
63 
64 int main(){
65     build(1,1,5);
66     print();
67     insert(1,1,4);
68     //insert(1,2,3); 
69     print();
70     //cout<<count(1)<<endl;
71     return 0;
72 } 
73 
74 void print(){
75     for(int i=1;i<=7;i++){
76         cout<<segTree[i].start<<" "<<segTree[i].end<<" "<<segTree[i].cover<<endl;
77     }
78     cout<<"-------------------------------------------------------------------------"<<endl;
79 }

 

 

再贴一份代码,上面那段代码insert位置有问题

 

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 struct node{
 4     int val;
 5     int start;
 6     int end;
 7 }tree[100];
 8 
 9 void print();
10 
11 void build(int root,int start,int end){
12     //这里是线段有没有覆盖,所以叶子节点是线段而不是点
13     tree[root].start=start;
14     tree[root].end=end;
15     tree[root].val=0;
16     if(start+1<end){
17         int mid=(start+end)/2;
18         build(2*root,start,mid);
19         build(2*root+1,mid,end);
20     } 
21 }
22 
23 void insert(int root,int left,int right){
24     if(tree[root].val==0){
25         int start=tree[root].start;
26         int end=tree[root].end;
27         int mid=(start+end)/2;
28         cout<<start<<" "<<end<<endl;
29         if(start==left&&end==right){
30             tree[root].val=1;
31         }
32         else if(start+1==end){
33             return ;
34         }
35         else if(mid<left){
36             insert(2*root+1,left,right);
37         }
38         else if(right<mid){
39             insert(2*root,left,right);
40         }
41         else{
42             insert(2*root,left,mid);
43             insert(2*root+1,mid,right);
44         }
45     }    
46 }
47 
48 int count(int root){
49     int start=tree[root].start;
50     int end=tree[root].end;
51     int val=tree[root].val;
52     if(val==1){
53         return end-start;
54     }
55     else if(end-start==1){
56         return 0;
57     }
58     else{
59         return count(root*2)+count(root*2+1);
60     }
61 }
62 
63 int main(){
64     build(1,1,5);
65     print();
66     insert(1,1,4);
67     print();
68     cout<<count(1)<<endl;
69     return 0;
70       
71 } 
72 
73 void print(){
74     for(int i=1;i<=7;i++){
75         cout<<tree[i].start<<" "<<tree[i].end<<" "<<tree[i].val<<endl;
76     }
77     cout<<endl;
78 }

 

posted @ 2017-09-11 23:27  范仁义  阅读(1945)  评论(0编辑  收藏  举报