#include <bits/stdc++.h>
namespace SegmentTreeLazy {
/*******************************************************************************
* SegmentTree<Value, Extra, Traits> - segment tree class with lazy propagation, 0-indexed
* Default operations: minimal value on segment and addition on segment for int64_t type
* Use Traits<Value,Extra> for definition of:
* 1) neutral element for `Value`;
* 2) neutral element for `Extra`;
* 3) how should combine `Extra` with `Value`;
* 4) how should combine `Value` with `Value` (children to root);
* 5) how should combine `Extra` with `Extra`;
* See examples below: TraitsMinAdd<Value, Extra>
******************************************************************************/
/*******************************************************************************
* Traits for minimal value on segment.
* Get-query: get minimal value in segment [l, r]
* Update-query: add const to each value in segment [l, r]
******************************************************************************/
template<typename Value, typename Extra>
struct TraitsMinAdd {
// Definition of neutral element for `Value`:
static Value valueNeutral() { return std::numeric_limits<Value>::max(); }
// Definition of neutral element for `Extra`:
static Extra extraNeutral() { return Extra(0); }
// Definition of how should combine `Extra` with `Value`:
template<typename Node>
static Value getValue(const Node& src) {
return src.value() + src.extra();
}
// Definition of how should combine `Value` with `Value` (children to root):
template<typename NodeRoot, typename NodeLt, typename NodeRt>
static void pull(NodeRoot root, const NodeLt& lt, const NodeRt& rt) {
root.value() = std::min(getValue(lt), getValue(rt));
}
// Definition of how should combine `Extra` with `Extra`:
template<typename NodeDst, typename NodeSrc>
static void push(NodeDst dst, const NodeSrc& src) {
dst.extra() += src.extra();
}
};
/*******************************************************************************
* Additional traits, implemented below
******************************************************************************/
template<typename Value, typename Extra> struct TraitsMaxAdd;
/*******************************************************************************
* SegmentTree, see description above
******************************************************************************/
template<typename Value = int64_t, typename Extra = int64_t, typename Traits = TraitsMinAdd<Value, Extra> >
struct SegmentTree {
/*******************************************************************************
* Node class
******************************************************************************/
struct Node {
Value value;
Extra extra;
Node(Value value_ = Traits::valueNeutral(), Extra extra_ = Traits::extraNeutral())
: value(value_), extra(extra_) { }
Value getValue(int l, int r) const { return Traits::getValue(NodeWrapper<Node>(l, r, *this)); }
};
/*******************************************************************************
* NodeWrapper class
******************************************************************************/
template<typename NodeType>
struct NodeWrapper {
int l, r;
NodeType node;
NodeWrapper(int l_, int r_, NodeType node_)
: l(l_), r(r_), node(node_) { }
int left() const { return l; }
int right() const { return r; }
int mid() const { return (l+r)/2; }
int len() const { return r - l + 1; }
Value& value() { return node.value; }
Extra& extra() { return node.extra; }
const Value& value() const { return node.value; }
const Extra& extra() const { return node.extra; }
};
/*******************************************************************************
* SegmentTree public data: n - number of items, data - vector for nodes
******************************************************************************/
int n; std::vector<Node> data;
/*******************************************************************************
* Resize segment tree data to needed size
******************************************************************************/
void resize(int n_) {
n = n_;
int pow = 1;
while (pow < n) { pow *= 2; }
data.assign(2 * pow, Node());
}
/*******************************************************************************
* Lazy propagation from node to its children
******************************************************************************/
void push(int v, int l, int r, int m) {
if (data[v].extra != Traits::extraNeutral()) {
Traits::push(
NodeWrapper<Node&>(l, m, data[2*v+1]),
NodeWrapper<const Node&>(l, r, data[v])
);
Traits::push(
NodeWrapper<Node&>(m+1, r, data[2*v+2]),
NodeWrapper<const Node&>( l, r, data[v])
);
data[v].extra = Traits::extraNeutral();
}
}
/*******************************************************************************
* Update node using children values
******************************************************************************/
void pull(int v, int l, int r, int m) {
assert(data[v].extra == Traits::extraNeutral());
Traits::pull(
NodeWrapper<Node&>( l, r, data[v]),
NodeWrapper<const Node&>( l, m, data[2*v+1]),
NodeWrapper<const Node&>(m+1, r, data[2*v+2])
);
}
/*******************************************************************************
* Build segtree from array with given values
******************************************************************************/
template<typename T>
void build(const std::vector<T>& arr, const int v, const int tl, const int tr) {
if (tl == tr) {
data[v] = Node(arr[tl]);
} else {
const int tm = (tl + tr) / 2;
build(arr, 2*v+1, tl, tm);
build(arr, 2*v+2, tm+1, tr);
pull(v, tl, tr, tm);
}
}
template<typename T>
void build(const std::vector<T>& arr) {
resize((int)arr.size());
build(arr, 0, 0, n-1);
}
/*******************************************************************************
* Get-query on range [ql, qr]
******************************************************************************/
Node get(int ql, int qr, const int v, const int tl, const int tr) {
if (ql == tl && qr == tr) {
return data[v];
} else {
int tm = (tl + tr) / 2;
push(v, tl, tr, tm);
Node ret;
if (qr <= tm) {
ret = get(ql, qr, 2*v+1, tl, tm);
} else if (ql > tm) {
ret = get(ql, qr, 2*v+2, tm+1, tr);
} else {
const auto lt = get( ql, tm, 2*v+1, tl, tm);
const auto rt = get(tm+1, qr, 2*v+2, tm+1, tr);
Traits::pull(
NodeWrapper<Node&>( ql, qr, ret),
NodeWrapper<const Node&>( ql, tm, lt),
NodeWrapper<const Node&>(tm+1, qr, rt)
);
}
pull(v, tl, tr, tm);
return ret;
}
}
Value get(const int ql, const int qr) { return get(ql, qr, 0, 0, n-1).getValue(ql, qr); }
/*******************************************************************************
* Update query on range [ql, qr] by extra
******************************************************************************/
void update(const int ql, const int qr, const Extra& extra, const int v, const int tl, const int tr) {
if (ql == tl && tr == qr) {
Traits::push(
NodeWrapper<Node&>(tl, tr, data[v]),
NodeWrapper<Node>(ql, qr, Node(Traits::valueNeutral(), extra))
);
} else {
int tm = (tl + tr) / 2;
push(v, tl, tr, tm);
if (qr <= tm) {
update(ql, qr, extra, 2*v+1, tl, tm);
} else if (ql > tm) {
update(ql, qr, extra, 2*v+2,tm+1,tr);
} else {
update(ql, tm, extra, 2*v+1, tl, tm);
update(tm+1, qr, extra, 2*v+2, tm+1, tr);
}
pull(v, tl, tr, tm);
}
}
void update(const int ql, const int qr, const Extra& extra) {
update(ql, qr, extra, 0, 0, n-1);
}
};
/*******************************************************************************
* Traits for maximal value on segment.
* Get-query: get maximal value in segment [l, r]
* Update-query: add const to each value in segment [l, r]
******************************************************************************/
template<typename Value, typename Extra>
struct TraitsMaxAdd {
// Definition of neutral element for `Value`:
static Value valueNeutral() { return std::numeric_limits<Value>::min(); }
// Definition of neutral element for `Extra`:
static Extra extraNeutral() { return Extra(0); }
// Definition of how should combine `Extra` with `Value`:
template<typename Node>
static Value getValue(const Node& src) {
return src.value() + src.extra();
}
// Definition of how should combine `Value` with `Value` (children to root):
template<typename NodeRoot, typename NodeLt, typename NodeRt>
static void pull(NodeRoot root, const NodeLt& lt, const NodeRt& rt) {
root.value() = std::max(getValue(lt), getValue(rt));
}
// Definition of how should combine `Extra` with `Extra`:
template<typename NodeDst, typename NodeSrc>
static void push(NodeDst dst, const NodeSrc& src) {
dst.extra() += src.extra();
}
};
}
int main() {
// Creating queries 1) arr[l..r] += x and 2) max(arr[l..r])
const int n = (int)1e6;
const int q = (int)1e6;
std::mt19937 gen;
std::uniform_int_distribution<int> dist(0,n-1);
std::vector<int> arr(n);
for (auto &it : arr) { it = dist(gen); }
std::vector<int> queryType(q), queryLeft(q), queryRight(q), queryExtra(q);
for (int i = 0; i < q; ++i) {
queryType[i] = 1 + dist(gen) % 2;
queryLeft[i] = dist(gen);
queryRight[i] = dist(gen);
if (queryLeft[i] > queryRight[i]) { std::swap(queryLeft[i], queryRight[i]); }
queryExtra[i] = dist(gen) - n / 2;
}
// Create SegmentTree
SegmentTreeLazy::SegmentTree<int64_t, int64_t, SegmentTreeLazy::TraitsMaxAdd<int64_t,int64_t>> segtree;
segtree.build(arr);
int64_t checkValue = 0;
for (int i = 0; i < q; ++i) {
if (queryType[i] == 1) {
segtree.update(queryLeft[i], queryRight[i], queryExtra[i]);
} else {
checkValue += segtree.get(queryLeft[i], queryRight[i]);
}
}
std::cout << "checkValue = " << checkValue << std::endl;
return 0;
}
#include <bits/stdc++.h>

namespace SegmentTreeLazy {
    
    /*******************************************************************************
     *  SegmentTree<Value, Extra, Traits> - segment tree class with lazy propagation, 0-indexed
     *  Default operations: minimal value on segment and addition on segment for int64_t type
     *  Use Traits<Value,Extra> for definition of:
     *      1)  neutral element for `Value`;
     *      2)  neutral element for `Extra`;
     *      3)  how should combine `Extra` with `Value`;
     *      4)  how should combine `Value` with `Value` (children to root);
     *      5)  how should combine `Extra` with `Extra`;
     *  See examples below: TraitsMinAdd<Value, Extra>
     ******************************************************************************/
    
    /*******************************************************************************
     *  Traits for minimal value on segment. 
     *  Get-query:    get minimal value in segment [l, r]
     *  Update-query: add const to each value in segment [l, r]
     ******************************************************************************/
    template<typename Value, typename Extra>
    struct TraitsMinAdd {
        // Definition of neutral element for `Value`:
        static Value valueNeutral() { return std::numeric_limits<Value>::max(); }
        // Definition of neutral element for `Extra`:
        static Extra extraNeutral() { return Extra(0); }
        // Definition of how should combine `Extra` with `Value`:
        template<typename Node>
        static Value getValue(const Node& src) {
            return src.value() + src.extra();
        }
        // Definition of how should combine `Value` with `Value` (children to root):
        template<typename NodeRoot, typename NodeLt, typename NodeRt>
        static void pull(NodeRoot root, const NodeLt& lt, const NodeRt& rt) {
            root.value() = std::min(getValue(lt), getValue(rt));
        }
        // Definition of how should combine `Extra` with `Extra`:
        template<typename NodeDst, typename NodeSrc>
        static void push(NodeDst dst, const NodeSrc& src) {
            dst.extra() += src.extra();
        }
    };
    
    /*******************************************************************************
     *  Additional traits, implemented below 
     ******************************************************************************/
    template<typename Value, typename Extra>  struct TraitsMaxAdd;
    
    /*******************************************************************************
     *  SegmentTree, see description above
     ******************************************************************************/
    template<typename Value = int64_t, typename Extra = int64_t, typename Traits = TraitsMinAdd<Value, Extra> >
    struct SegmentTree {
        
        /*******************************************************************************
         *  Node class
         ******************************************************************************/
        struct Node {
            Value value;
            
            Extra extra;
            
            Node(Value value_ = Traits::valueNeutral(), Extra extra_ = Traits::extraNeutral())
                : value(value_), extra(extra_) { }
            
            Value getValue(int l, int r) const { return Traits::getValue(NodeWrapper<Node>(l, r, *this)); }
        };
        
        /*******************************************************************************
         *  NodeWrapper class
         ******************************************************************************/
        template<typename NodeType>
        struct NodeWrapper {
            int l, r;
            NodeType node;
            NodeWrapper(int l_, int r_, NodeType node_)
                : l(l_), r(r_), node(node_) { }
            int  left() const { return l; }
            int right() const { return r; }
            int   mid() const { return (l+r)/2; }
            int   len() const { return r - l + 1; }
            Value& value() { return node.value; }
            Extra& extra() { return node.extra; }
            const Value& value() const { return node.value; }
            const Extra& extra() const { return node.extra; }
        };
        
        /*******************************************************************************
         *  SegmentTree public data: n - number of items, data - vector for nodes
         ******************************************************************************/
        int n; std::vector<Node> data;
        
        
        /*******************************************************************************
         *  Resize segment tree data to needed size
         ******************************************************************************/
        void resize(int n_) {
            n = n_;
            int pow = 1;
            while (pow < n) { pow *= 2; }
            data.assign(2 * pow, Node());
        }
        
        /*******************************************************************************
         *  Lazy propagation from node to its children
         ******************************************************************************/
        void push(int v, int l, int r, int m) {
            if (data[v].extra != Traits::extraNeutral()) {
                Traits::push(
                    NodeWrapper<Node&>(l, m, data[2*v+1]), 
                    NodeWrapper<const Node&>(l, r, data[v])
                );
                Traits::push(
                    NodeWrapper<Node&>(m+1, r, data[2*v+2]), 
                    NodeWrapper<const Node&>(  l, r, data[v])
                );
                data[v].extra = Traits::extraNeutral();
            }
        }
        
        /*******************************************************************************
         *  Update node using children values
         ******************************************************************************/
        void pull(int v, int l, int r, int m) {
            assert(data[v].extra == Traits::extraNeutral());
            Traits::pull(
                NodeWrapper<Node&>(  l, r, data[v]), 
                NodeWrapper<const Node&>(  l, m, data[2*v+1]), 
                NodeWrapper<const Node&>(m+1, r, data[2*v+2])
            );
        }
        
        /*******************************************************************************
         *  Build segtree from array with given values
         ******************************************************************************/
        template<typename T>
        void build(const std::vector<T>& arr, const int v, const int tl, const int tr) {
            if (tl == tr) {
                data[v] = Node(arr[tl]);
            } else {
                const int tm = (tl + tr) / 2;
                build(arr, 2*v+1,   tl, tm);
                build(arr, 2*v+2, tm+1, tr);
                pull(v, tl, tr, tm);
            }
        }
        
        template<typename T>
        void build(const std::vector<T>& arr) { 
            resize((int)arr.size());
            build(arr, 0, 0, n-1);
        }

        /*******************************************************************************
         *  Get-query on range [ql, qr]
         ******************************************************************************/
        Node get(int ql, int qr, const int v, const int tl, const int tr) {
            if (ql == tl && qr == tr) {
                return data[v];
            } else {
                int tm = (tl + tr) / 2;
                push(v, tl, tr, tm);
                Node ret;
                if (qr <= tm) {
                    ret = get(ql, qr, 2*v+1,   tl, tm);
                } else if (ql > tm) {
                    ret = get(ql, qr, 2*v+2, tm+1, tr);
                } else {
                    const auto lt = get(  ql, tm, 2*v+1,   tl, tm);
                    const auto rt = get(tm+1, qr, 2*v+2, tm+1, tr);
                    Traits::pull(
                        NodeWrapper<Node&>(  ql, qr, ret), 
                        NodeWrapper<const Node&>(  ql, tm, lt), 
                        NodeWrapper<const Node&>(tm+1, qr, rt)
                    );
                }
                pull(v, tl, tr, tm);
                return ret;
            }
        }
        
        Value get(const int ql, const int qr) { return get(ql, qr, 0, 0, n-1).getValue(ql, qr); }
        
        /*******************************************************************************
         *  Update query on range [ql, qr] by extra
         ******************************************************************************/
        void update(const int ql, const int qr, const Extra& extra, const int v, const int tl, const int tr) {
            if (ql == tl && tr == qr) {
                Traits::push(
                    NodeWrapper<Node&>(tl, tr, data[v]),
                    NodeWrapper<Node>(ql, qr, Node(Traits::valueNeutral(), extra))
                );
            } else {
                int tm = (tl + tr) / 2;
                push(v, tl, tr, tm);
                if (qr <= tm) {
                    update(ql, qr, extra, 2*v+1, tl, tm);
                } else if (ql > tm) {
                    update(ql, qr, extra, 2*v+2,tm+1,tr);
                } else {
                    update(ql, tm, extra, 2*v+1,   tl, tm);
                    update(tm+1, qr, extra, 2*v+2, tm+1, tr);
                }
                pull(v, tl, tr, tm);
            }
        }

        void update(const int ql, const int qr, const Extra& extra) {
            update(ql, qr, extra, 0, 0, n-1); 
        }

    };
    
    /*******************************************************************************
     *  Traits for maximal value on segment. 
     *  Get-query:    get maximal value in segment [l, r]
     *  Update-query: add const to each value in segment [l, r]
     ******************************************************************************/
    template<typename Value, typename Extra>
    struct TraitsMaxAdd {
        // Definition of neutral element for `Value`:
        static Value valueNeutral() { return std::numeric_limits<Value>::min(); }
        // Definition of neutral element for `Extra`:
        static Extra extraNeutral() { return Extra(0); }
        // Definition of how should combine `Extra` with `Value`:
        template<typename Node>
        static Value getValue(const Node& src) {
            return src.value() + src.extra();
        }
        // Definition of how should combine `Value` with `Value` (children to root):
        template<typename NodeRoot, typename NodeLt, typename NodeRt>
        static void pull(NodeRoot root, const NodeLt& lt, const NodeRt& rt) {
            root.value() = std::max(getValue(lt), getValue(rt));
        }
        // Definition of how should combine `Extra` with `Extra`:
        template<typename NodeDst, typename NodeSrc>
        static void push(NodeDst dst, const NodeSrc& src) {
            dst.extra() += src.extra();
        }
    };   
}

int main() {
    // Creating queries 1) arr[l..r] += x and 2) max(arr[l..r])
    const int n = (int)1e6;
    const int q = (int)1e6;
    std::mt19937 gen;
    std::uniform_int_distribution<int> dist(0,n-1);
    std::vector<int> arr(n);
    for (auto &it : arr) { it = dist(gen); }
    std::vector<int> queryType(q), queryLeft(q), queryRight(q), queryExtra(q);
    for (int i = 0; i < q; ++i) {
        queryType[i] = 1 + dist(gen) % 2;
        queryLeft[i] = dist(gen);
        queryRight[i] = dist(gen);
        if (queryLeft[i] > queryRight[i]) { std::swap(queryLeft[i], queryRight[i]); }
        queryExtra[i] = dist(gen) - n / 2;
    }
    // Create SegmentTree
    SegmentTreeLazy::SegmentTree<int64_t, int64_t, SegmentTreeLazy::TraitsMaxAdd<int64_t,int64_t>> segtree;
    segtree.build(arr);
    int64_t checkValue = 0;
    for (int i = 0; i < q; ++i) {
        if (queryType[i] == 1) {
            segtree.update(queryLeft[i], queryRight[i], queryExtra[i]);
        } else {
            checkValue += segtree.get(queryLeft[i], queryRight[i]);
        }
    }
    std::cout << "checkValue = " << checkValue << std::endl;
    return 0;
}