线段树入门学习(二)(兼解POJ3468) JAVA

继续上文[url]http://128kj.iteye.com/blog/1738772[/url]:
在那里用链状结构实现了二叉线段树,下面程序使用一维数组以完全二叉树的方式来存储。

先看一维数组存储线段树到底需要开多大的数组,网上有一个计算程序:
[code="java"]
import java.util.Scanner;
/*线段树空间计算程序 Power By:Comzyh*/

class segment {//线段树节点
int b,e;
}

public class SegmentTree{//线段树,用数组实现
static int M=5000000;
segment seg[];

int Nnode;//节点数
int LastNode;//最后一个节点

public SegmentTree(){
seg=new segment[M];
for(int i=0;i<M;i++)
seg[i]=new segment();
}


void build(int b, int e, int root){//构造线段树
Nnode++;
if (root>LastNode)
LastNode=root;
seg.b=b;
seg.e=e;
if (b==e)
return;
int mid =(b+e)>>1;
build(b,mid,root<<1);
build(mid+1,e,(root<<1)+1);
}

public int getNnode(){
return Nnode;
}

public int getLastNode(){
return LastNode;
}

public static void main(String args[]){
Scanner in=new Scanner(System.in);
int N;
while (true){
System.out.printf("请输入区间长度:\n");
N=in.nextInt();
if (N==0) break;
SegmentTree st=new SegmentTree();
st.build(1,N,1);
System.out.printf("线段树构造完成, 共有%d 节点,最后一个节点是: %d\n",st.getNnode(),st.getLastNode());
//注意:节点个数总是2N-1
}
}
}
[/code]

运行:
C:\ex>java SegmentTree
请输入区间长度:
5
线段树构造完成, 共有9 节点,最后一个节点是: 9
请输入区间长度:
10
线段树构造完成, 共有19 节点,最后一个节点是: 25
请输入区间长度:
100000
线段树构造完成, 共有199999 节点,最后一个节点是: 262141

对下面的程序,数组开到300000就可以了。

例:POJ3468

大致题意:给出一个长度为n的整数序列。在这个序列上有两个操作。
其中:
“C a b c”表示区间[a,b]里的所有数都增加c。

“Q a b”表示求出区间[a,b]里的所有数的和。

思路: 线段树+lazy思想:
样例:
Sample Input

10 5
1 2 3 4 5 6 7 8 9 10
Q 4 4
Q 1 10
Q 2 4
C 3 6 3
Q 2 4

Sample Output

4
55
9
15

下面是AC过的代码:

import java.util.Scanner;
import java.io.BufferedInputStream;
class Tree{ //线段树节点
int l, r;
long v, add;
}
public class Main{//数组实现的二叉线段树

static int N= 100005;
Tree node[]=new Tree[N*3];
long sum[];


public Main(long[] sum){
this.sum=sum;
for(int i=0;i<3*N;i++)
node[i]=new Tree();
}



public int L(int u){
return u << 1;
}

public int R(int u) {
return u << 1 | 1 ;
}

public void build ( int u, int l, int r ) //建以r为根的线段树[l,r]
{
node[u].l = l;
node[u].r = r;
node[u].add = 0;
node[u].v = sum[r] - sum[l-1];
if ( l == r ) return;
int mid = ( l + r ) >> 1;
build ( L(u), l, mid );
build ( R(u), mid+1, r );
}

public void update ( int u, int l, int r, int val ) //更新
{
if ( l == node[u].l && node[u].r == r )
{
node[u].add += val;
node[u].v += val * ( r - l + 1 );
return;
}

if ( node[u].add != 0 )
{
node[L(u)].add += node[u].add;
node[L(u)].v += node[u].add * ( node[L(u)].r - node[L(u)].l + 1 );
node[R(u)].add += node[u].add;
node[R(u)].v += node[u].add * ( node[R(u)].r - node[R(u)].l + 1 );
node[u].add = 0;
}

int mid = ( node[u].l + node[u].r ) >> 1;
if ( r <= mid )
update ( L(u), l, r, val );
else if ( l > mid )
update ( R(u), l, r, val );
else
{
update ( L(u), l, mid, val );
update ( R(u), mid+1, r, val );
}
node[u].v = node[L(u)].v + node[R(u)].v;
}

public long query ( int u, int l, int r ) //查询
{
if ( l == node[u].l && node[u].r == r )
return node[u].v;

if ( node[u].add != 0 )
{
node[L(u)].add += node[u].add;
node[L(u)].v += node[u].add * ( node[L(u)].r - node[L(u)].l + 1 );
node[R(u)].add += node[u].add;
node[R(u)].v += node[u].add * ( node[R(u)].r - node[R(u)].l + 1 );
node[u].add = 0;
}

int mid = ( node[u].l + node[u].r ) >> 1; // 计算左右子结点的分隔点
if ( r <= mid)
return query ( L(u), l, r ); // 和左孩子有交集,考察左子结点
else if ( l > mid )
return query ( R(u), l, r ); // 和右孩子有交集,考察右子结点
else
return query ( L(u), l, mid ) + query ( R(u), mid+1, r );
}
public static void main(String args[]){

Scanner in = new Scanner(new BufferedInputStream(System.in));


int n = in.nextInt();
int m = in.nextInt();

long[] sum=new long[n+1];
for(int i = 1; i<=n; i++){
sum[i] = sum[i-1] + in.nextInt();
}

Main st=new Main(sum);
st.build(1,1,n);
//st.printTree(1);
String cmd;
int x;
int y;
int c;
for(int i = 0; i<m; i++)
{
cmd = in.next();
if (cmd.equals("Q")) {
x = in.nextInt();
y = in.nextInt();
System.out.println(st.query(1,x, y));
} else {
x = in.nextInt();
y = in.nextInt();
c=in.nextInt();
// System.out.println("c="+c);
st.update(1,x,y,c);
}
}

}

}


样例2:

10 22
1 2 3 4 5 6 7 8 9 10
Q 4 4
C 1 10 3
C 6 10 3
C 6 9 3
C 8 9 -100
C 7 9 3
C 7 10 3
C 1 10 3
Q 6 10
Q 6 9
Q 8 9
Q 7 9
Q 7 10
Q 1 10
Q 2 4
C 3 6 3
Q 9 9
Q 1 1
Q 5 5
Q 6 6
Q 7 7
Q 6 8

ans

4
-82
-104
-147
-122
-100
-37
27
-73
7
14
21
25
-28

源码下载:
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值