c++ - 在线段树中查询

标签 c++ algorithm data-structures tree segment-tree

我一直在努力理解 query() 函数中最后 6 行的逻辑。

这是问题的代码 GSS1在 spoj 上。

解决方案 link

#include <cstdio>
#include <algorithm>
#define MAX 70000

using namespace std;

struct no {
int lsum, rsum, msum;
};

int array[ MAX + 1 ], sums[ MAX + 1 ];
no tree[ 4 * MAX + 1 ];

 void init( int node, int i, int j ) {
if ( i == j ) {
    tree[ node ] = ( ( no ) { array[ i ], array[ i ], array[ i ] } );
}
else {
    init( node * 2, i, ( i + j ) / 2 );
    init( node * 2 + 1, ( i + j ) / 2 + 1, j );
    no left = tree[ node * 2 ], right = tree[ node * 2 + 1 ];
    tree[ node ].lsum = max( left.lsum, sums[ ( i + j ) / 2 ] - sums[ i - 1 ] + right.lsum );
    tree[ node ].rsum = max( right.rsum, sums[ j ] - sums[ ( i + j ) / 2 ] + left.rsum );
    tree[ node ].msum = max( left.msum, max( right.msum, left.rsum + right.lsum ) );
}}

 no query( int node, int a, int b, int i, int j ) {
if ( a == i && b == j ) {
    return tree[ node ];
}
else if ( j <= ( a + b ) / 2 ) {
    return query( node * 2, a, ( a + b ) / 2, i, j );
}
if ( i > ( a + b ) / 2 ) {
    return query( node * 2 + 1, ( a + b ) / 2 + 1, b, i, j );
}
no left = query( node * 2, a, ( a + b ) / 2, i, ( a + b ) / 2 );
no right = query( node * 2 + 1, ( a + b ) / 2 + 1, b, ( a + b ) / 2 + 1, j );
return ( ( no ) {
            max( left.lsum, sums[ ( a + b ) / 2 ] - sums[ i - 1 ] + right.lsum ),
            max( right.rsum, sums[ b ] - sums[ ( a + b ) / 2 ] + left.rsum ),
            max( left.msum, max( right.msum, left.rsum + right.lsum ) )
            } ); }

int main() {
int i, N, q, l, r;
scanf( "%d", &N );
for ( i = 0; i < N; ++i ) {
    scanf( "%d", array + i );
    if ( i == 0 ) {
        sums[ i ] = array[ i ];
    }
    else {
        sums[ i ] = sums[ i - 1 ] + array[ i ];
    }
}
init( 1, 0, N - 1 );
scanf( "%d", &q );
for ( i = 0; i < q; ++i ) {
    scanf( "%d%d", &l, &r );
    --l;
    --r;
    printf( "%d\n", query( 1, 0, N - 1, l, r ).msum );
}
return 0; }

查询函数中的 no = left && no = right 和 return 需要什么?

有没有人建议更好的实现/教程 fr 线段树。

在实现数据结构时,我无法想象这些递归。有什么建议吗?

最佳答案

What is the need of that no = left && no = right and that return in query function

这就是您查询的段进入两个子段的情况。

假设您有一个覆盖区间 1..n 的节点并且有 2 个子节点 1 .. n/2 和 n/2+1 ..n。如果你想查询某个区间 [a,b] 使得 a <=n/2 < b 那么你需要从左分支和右分支获取结果并将它们组合(换句话说将查询拆分为 [a , n/2] 和 [n/2 +1, b] 得到2个结果并合并。

为了提高效率,您可以证明对于任何查询都只能有一个这样的拆分,因为间隔现在已经触及边缘(因此,虽然您总是向下遍历其中一个子节点,但另一个子节点要么被忽略,要么完全落入查询并在下一个递归步骤返回)。

该代码用于非常有效的实现(您不存储指向子节点的指针,节点范围是隐式的)。如果您正在尝试学习/调试,请寻找可以更明确地存储内容的东西,或者自己编写一个。至少正确地缩进代码,将变量名更改为更易读的名称,并将 (a+b)/2 替换为 middle。这应该使代码更容易理解。还要在纸上画出结构(总是有帮助)。

关于c++ - 在线段树中查询,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36417338/

相关文章:

c++ - 给定一个无符号整数,很容易交换每个偶数位和奇数位。这可以概括为交换每个偶数和奇数 n 位吗?

c++ - 如何使用 2 个主要 src 文件构建 cmake 项目

algorithm - 图调度交集

python - 如何在 url 列表中快速找到不返回 302(重定向)状态代码的最后一个可用 url

algorithm - 有效地在双向链表中搜索具有指针约束的值?

c++ - 如何在 .cpp 文件中使用 Cuda 数据结构

c++ - 类函数找不到类定义的变量

c++ - 基类和派生类的二维动态数组

data-structures - 循环队列的缺点?

在数组中查找三个数字的算法,使得 a< b < c 和 v[a]<v[c]<v[b]