public class NumArray
{
    public interface Merger<T>
    {
        T morge(T a, T b);

    }

    public class SegmentTree<T>
    {
        private T[] data;
        public T[] tree;
        private Merger<T> merger;
        public SegmentTree(T[] arr, Merger<T> merger)
        {
            this.merger = merger;
            data = new T[arr.Length];
            for (var i = 0; i < arr.Length; i++)
            {
                data[i] = arr[i];
            }
            tree = new T[4 * arr.Length];
            buildSegmentTree(0, 0, data.Length - 1);
        }

        private void buildSegmentTree(int treeindex, int l, int r)
        {
            if (l == r)
            {
                tree[treeindex] = data[l];
                return;
            }
            int leftTreeIndex = leftChild(treeindex);
            int rightTreeIndex = rightChild(treeindex);
            int mid = l + (r - l) / 2;
            buildSegmentTree(leftTreeIndex, l, mid);
            buildSegmentTree(rightTreeIndex, mid + 1, r);

            tree[treeindex] = merger.morge(tree[leftTreeIndex], tree[rightTreeIndex]);

        }

        public int getSize()
        {
            return data.Length;
        }
        public T get(int index)
        {
            if (index < 0 || index >= data.Length)
                throw new System.Exception("Index illegal");
            return data[index];
        }
        public int leftChild(int index)
        {
            return 2 * index + 1;

        }
        public int rightChild(int index)
        {
            return 2 * index + 2;
        }
        public override string ToString()
        {

            System.Text.StringBuilder stringBuilder = new System.Text.StringBuilder();
            stringBuilder.Append('[');
            for (var i = 0; i < tree.Length; i++)
            {
                if (tree[i] != null)
                {
                    stringBuilder.Append(tree[i]);
                }
                else
                {
                    stringBuilder.Append("null");
                }
                stringBuilder.Append(",");
            }
            stringBuilder.Append("]");
            return stringBuilder.ToString();
        }
        /// <summary>
        /// 返回区间 【queryL queryR】的值
        /// </summary>
        /// <param name="queryl"></param>
        /// <param name="queryR"></param>
        /// <returns></returns>
        public T query(int queryl, int queryR)
        {
            if (queryl < 0 || queryl >= data.Length || queryR < 0 || queryR >= data.Length || queryl > queryR)
                throw new System.Exception("index is illegil");
            return query(0, 0, data.Length - 1, queryl, queryR);
        }
        private T query(int treeindex, int l, int r, int queryl, int queryR)
        {
            if (l == queryl && r == queryR)
            {
                return tree[treeindex];
            }
            int mid = l + (r - l) / 2;
            int leftIndex = leftChild(treeindex);
            int rightIndex = rightChild(treeindex);
            if (queryl >= mid + 1)
            {
                return query(rightIndex, mid + 1, r, queryl, queryR);
            }
            else if (queryR <= mid)
            {
                return query(leftIndex, l, mid, queryl, queryR);
            }
            T leftResult = query(leftIndex, l, mid, queryl, mid);
            T rightResult = query(rightIndex, mid + 1, r, mid + 1, queryR);
            return merger.morge(leftResult, rightResult);
        }
        public void set(int index, T data)
        {
            if (index < 0 || index >= this.data.Length)
            {
                throw new System.Exception(" index is illegal");
            }
            set(0, 0, this.data.Length - 1, index, data);
        }
        private void set(int treeIndex, int l, int r, int index, T data)
        {
            if (l == r)
            {
                tree[treeIndex] = data;
                return;
            }
            int mid = l + (r - l) / 2;
            int leftTreeIndex = leftChild(treeIndex);
            int rightTreeIndex = rightChild(treeIndex);
            if (index >= mid + 1)
            {
                set(rightTreeIndex, mid + 1, r, index, data);
            }
            else
            {
                set(leftTreeIndex, l, mid, index, data);
            }
            tree[treeIndex] = merger.morge(tree[leftTreeIndex], tree[rightTreeIndex]);
        }
    }
    class MyMerger<T> : Merger<T>
    {
        public T morge(T a, T b)
        {
            dynamic v1 = a;
            dynamic v2 = b;
            return (T)(v1 + v2);
        }
    }
    SegmentTree<int> tree;
    public NumArray(int[] nums)
    {
        tree = new SegmentTree<int>(nums, new MyMerger<int>());
    }

    public void Update(int index, int val)
    {
        if (tree == null) throw new System.Exception("....");
        tree.set(index, val);
    }

    public int SumRange(int left, int right)
    {
        if (tree == null) throw new System.Exception("....");
        return tree.query(left, right);
    }
}

/**
 * Your NumArray object will be instantiated and called as such:
 * NumArray obj = new NumArray(nums);
 * obj.Update(index,val);
 * int param_2 = obj.SumRange(left,right);
 */