void propagate(node_type *p) const { if(p == nullptr){ return; } const size_t k = size(p->children[0]); const auto mod_l_cr = m_traits.split_modifier(p->modifier, k); const auto mod_c_r = m_traits.split_modifier(mod_l_cr.second, 1); if(p->children[0] != nullptr){ p->children[0]->modifier = m_traits.merge_modifier( p->children[0]->modifier, mod_l_cr.first); refresh(p->children[0]); } if(p->children[1] != nullptr){ p->children[1]->modifier = m_traits.merge_modifier( p->children[1]->modifier, mod_c_r.second); refresh(p->children[1]); } p->value = m_traits.resolve(1, p->value, mod_c_r.first); p->modifier= m_traits.default_modifier(); }
/** * @brief 区間更新 * * インデックスが区間 [l, r) に含まれる要素すべてxに従って更新する。 * - 時間計算量: \f$ O(\log{n}) \f$ * * @param[in] l 区間の始端 * @param[in] r 区間の終端 * @param[in] x 更新クエリ */ void modify(size_t l, size_t r, const modifier_type &x){ const auto p0 = split(m_root, l); const auto p1 = split(p0.second, r - l); node_type *c = p1.first; if(c != nullptr){ c->modifier = m_traits.merge_modifier(c->modifier, x); refresh(c); } m_root = merge(merge(p0.first, c), p1.second); }