线段树的作用

线段树是一种二叉搜索树,它能以较低时间复杂度 O(logN) 得解决大规模求和、gcd 等
空间复杂度为 2N,但实际应用时还需 4N 的空间

查询和单点更改的代码实现

class NumArray
{
  public:
    int *tree;
    int size;

    NumArray(vector<int> nums)
    {
        size = nums.size();
        //size 储存原始数据的数量
        tree = new int[size * 4];
        if (size != 0)
            Build(1, size, 1, nums);
        //以 1 为根节点,为了防止计算问题而废除 0 号位置
    }

线段树大小为原始数组大小的 4 倍

==>Q:树叶节点和树节点一共 2n 个数据,为什么需要 4 倍?
==>A:在递归时候仍然需要访问叶子节点下一层子树,下一层子节点一共有 2n 个,所以共计 4n 的空间需求

void Build(int left, int right, int index, vector<int> &nums)
{
    if (left == right) //如果到达叶子节点不再向下访问
    {
        tree[index] = nums[left - 1];
        //==>Q:为什么需要访问 nums[left-1] 而不是 nums[left]?
        //==>A:tree 数组以 1 位开始为第一个数据,而传入的数组是以 0 位为第一个数据
        return;
    }
    int mid = (left + right) / 2;
    Build(left, mid, 2 * index, nums);
    Build(mid + 1, right, 2 * index + 1, nums);
    PushUp(index);
    //建完左右子树后更新自己的数据
}

==>Q:为什么左右节点的序号分别是 2n 和 2n+1?
==>A:见下图

               1
              (1)
        2             3
      (2*1)        (2*1+1)
    4       5     6       7
  (2*2)  (2*2+1)(2*3)  (2*3+1)

…continue

这种编号顺序无浪费而且易知根节点为 index/2

void PushUp(int index)
{
    tree[index] = tree[2 * index] + tree[2 * index + 1];
    //根数据=左数据 + 右数据
}

void Update(int position, int val, int left, int right, int index)
{
    if (left == right) //如果到达叶子节点不再向下访问
    {
        tree[index] = val;
        //叶子节点值更改
        return;
    }
    int mid = (left + right) / 2;
    if (position <= mid) //算是一种二分查找,需要更改的位置在二分后是中值的左或者右边
        Update(position, val, left, mid, index * 2);
    //如果更改在中值左侧就访问左子树
    else
        Update(position, val, mid + 1, right, index * 2 + 1);
    //如果更改在中值右侧就访问右子树
    PushUp(index);
    //更新自己的数据,使更改一层层加起来
}

int Query(int Left, int Right, int left, int right, int index)
{
    if (Left <= left && right <= Right)
    //如果数据值在所求区域内....[Left..[left.......right]......Right]....
    {
        return tree[index];
        //返回我的数据,不再递归
    }
    int mid = (left + right) / 2;
    int ans = 0;
    if (Left <= mid) //如果中值>=所求区域左临界
        ans += Query(Left, Right, left, mid, index * 2);
    if (Right >= mid + 1) //如果中值 +1<=所求区域右临界
        ans += Query(Left, Right, mid + 1, right, index * 2 + 1);
    return ans;
}
                  [1  2  3  4  5  6  7  8  9  10  11  12  13]   => 求[4,12]
                  [1  2  3  4  5  6  7][8  9  10  11  12  13]
                  [1  2  3  4][5  6  7][8  9  10][11  12  13]
                        [3  4]   XXX      XXX    [11  12]
                           [4]                     XXX
                           XXX

XXX 表示返回上一个节点的数据不再递归
[4,12]==>[4] + [5,7] + [8,10] + [11,12]
在[1 2 3 4]被拆分为[1 2][3 4]时由于 mid=2<3 故不再访问左子树[1 2]
[13],[3]同理
在[1 2 3 4 5 6 7]被拆分为[5 6 7]时由于 5>3 && 8<12 则直接返回其数据不再递归
[8 9 10 11]同理

    void update(int i, int val)
    {
        Update(i + 1, val, 1, size, 1);
        //程序查询时候从 0 开始计位,故需要加一,从 [1,size] 中递归更改值
    }

    int sumRange(int i, int j)
    {
        return Query(i + 1, j + 1, 1, size, 1);
        //程序查询时候从 0 开始计位,故需要加一,从 [1,size] 中递归查找结果值
    }
};