class NgLib:: DynamicRangeSum(T)

Overview

動的な数列 $a$ に対して累積和を求めます。

累積和クエリは $O(\log{N})$ で処理することができます。

実装は BIT (Fenwick Tree) です。

もし、区間加算 $1$ 点取得がしたい場合は、このライブラリといもす法を組み合わせると良いです。 長さ $n$ の数列に対して操作する場合、BIT の長さは $n + 1$ 必要なことに注意してください。

Defined in:

nglib/data_structure/dynamic_range_sum.cr

Constructors

Instance Method Summary

Constructor Detail

def self. new (n : Int , val : T) #

長さが $n$ で各要素が $val$ の数列 $a$ を構築します。


[ View source ]
def self. new (elems : Enumerable(T)) #

長さが $n$ で $i$ 番目の要素が $elems_i$ の数列 $a$ を構築します。


[ View source ]
def self. new (n : Int ) #

長さが $n$ で各要素が $0$ の数列 $a$ を構築します。


[ View source ]

Instance Method Detail

def [] (l, r) : T #

$[l, r)$ 番目までの要素の総和 $\sum_{i=l}^{r-1} a_i$ を $O(\log{N})$ で返します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum[0, 5] # => 1 + 1 + 2 + 3 + 5 = 12

[ View source ]
def [] (range : Range( Int | Nil, Int | Nil)) : T #

range の表す範囲の要素の総和 $\sum_{i \in range} a_i$ を $O(\log{N})$ で返します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum[0...5] # => 1 + 1 + 2 + 3 + 5 = 12

[ View source ]
def [] (i) : T #

$a_i$ を $O(\log{N})$ で返します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum[5] # => 8

[ View source ]
def []= (i : Int , x : T) : T #

$a_i$ の値を $x$ に更新します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum.get?(0...5) # => 1 + 1 + 2 + 3 + 5 = 12
csum[0] = 100
csum.get?(0...5) # => 100 + 1 + 2 + 3 + 5 = 111

[ View source ]
def []? (l, r) : T | Nil #

$[l, r)$ 番目までの要素の総和 $\sum_{i=l}^{r-1} a_i$ を $O(\log{N})$ で返します。

$0 \leq l \leq r \leq n$ を満たさないとき、 nil を返します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum[0, 5]? # => 1 + 1 + 2 + 3 + 5 = 12
csum[7, 3]? # => nil

[ View source ]
def []? (range : Range( Int | Nil, Int | Nil)) : T #

range の表す範囲の要素の総和 $\sum_{i \in range} a_i$ を $O(\log{N})$ で返します。

$0 \leq l \leq r \leq n$ を満たさないとき、 nil を返します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum[0...5]? # => 1 + 1 + 2 + 3 + 5 = 12
csum[7...3]? # => nil

[ View source ]
def []? (i) : T | Nil #

$a_i$ を $O(\log{N})$ で返します。

$0 \leq i \lt n$ を満たさないとき、 nil を返します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum[5]   # => 8
csum[10]? # => nil

[ View source ]
def add (i : Int , x : T) #

$a_i$ に $x$ を加算します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum.get?(0...5) # => 1 + 1 + 2 + 3 + 5 = 12
csum.add(0, 99)
csum.get?(0...5) # => 100 + 1 + 2 + 3 + 5 = 111

[ View source ]
def get (l, r) : T #

$[l, r)$ 番目までの要素の総和 $\sum_{i=l}^{r-1} a_i$ を $O(\log{N})$ で返します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum.get(0, 5) # => 1 + 1 + 2 + 3 + 5 = 12

[ View source ]
def get (range : Range( Int | Nil, Int | Nil)) : T #

range の表す範囲の要素の総和 $\sum_{i \in range} a_i$ を $O(\log{N})$ で返します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum.get(0...5) # => 1 + 1 + 2 + 3 + 5 = 12

[ View source ]
def get (i) : T #

$a_i$ を $O(\log{N})$ で返します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum.get(5) # => 8

[ View source ]
def get? (l, r) : T | Nil #

$[l, r)$ 番目までの要素の総和 $\sum_{i=l}^{r-1} a_i$ を $O(\log{N})$ で返します。

$0 \leq l \leq r \leq n$ を満たさないとき、 nil を返します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum.get?(0, 5) # => 1 + 1 + 2 + 3 + 5 = 12
csum.get?(7, 3) # => nil

[ View source ]
def get? (range : Range( Int | Nil, Int | Nil)) : T #

range の表す範囲の要素の総和 $\sum_{i \in range} a_i$ を $O(\log{N})$ で返します。

$0 \leq l \leq r \leq n$ を満たさないとき、 nil を返します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum.get?(0...5) # => 1 + 1 + 2 + 3 + 5 = 12
csum.get?(7...3) # => nil

[ View source ]
def get? (i) : T | Nil #

$a_i$ を $O(\log{N})$ で返します。

$0 \leq i \lt n$ を満たさないとき、 nil を返します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum.get?(5) # => 8
csum.get?(10)? # => nil

[ View source ]
def set (i : Int , x : T) #

$a_i$ の値を $x$ に更新します。

array = [1, 1, 2, 3, 5, 8, 13]
csum = DynamicRangeSum(Int32).new(array)
csum.get?(0...5) # => 1 + 1 + 2 + 3 + 5 = 12
csum.set(0, 100)
csum.get?(0...5) # => 100 + 1 + 2 + 3 + 5 = 111

[ View source ]
def size : Int32 #

[ View source ]