procon_lib_rs

This documentation is automatically generated by online-judge-tools/verification-helper

View the Project on GitHub CoCo-Japan-pan/procon_lib_rs

:heavy_check_mark: crates/data_structure/lazy_segtree/src/lib.rs

Depends on

Required by

Verified with

Code

//! <https://github.com/rust-lang-ja/ac-library-rs/blob/master/src/lazysegtree.rs> を基にしています  
//! compositionやmappingに可変参照を用いているところと、作用が可変なら伝播を一部サボる部分が異なる

use algebra::{Action, ActionMonoid, Commutative, Monoid};
use internal_bits::ceil_log2;
use std::ops::{Bound::*, RangeBounds};

#[derive(Debug)]
pub struct LazySegTree<F: ActionMonoid> {
    range_size: usize,
    leaf_size: usize,
    log: usize,
    data: Vec<<F::M as Monoid>::Target>,
    lazy: Vec<F::A>,
}

impl<F: ActionMonoid> From<Vec<<F::M as Monoid>::Target>> for LazySegTree<F> {
    fn from(mut v: Vec<<F::M as Monoid>::Target>) -> Self {
        let range_size = v.len();
        let log = ceil_log2(range_size as u32) as usize;
        let leaf_size = 1 << log;
        let mut data = vec![F::M::id_element(); leaf_size];
        data.reserve(leaf_size);
        data.append(&mut v);
        data.append(&mut vec![F::M::id_element(); leaf_size - range_size]);
        let lazy = vec![F::A::id_action(); leaf_size];
        let mut ret = Self {
            range_size,
            leaf_size,
            log,
            data,
            lazy,
        };
        for i in (1..leaf_size).rev() {
            ret.update(i);
        }
        ret
    }
}

impl<F: ActionMonoid> LazySegTree<F> {
    pub fn new(n: usize) -> Self {
        vec![F::M::id_element(); n].into()
    }

    pub fn set(&mut self, mut p: usize, x: <F::M as Monoid>::Target) {
        assert!(p < self.range_size);
        p += self.leaf_size;
        for i in (1..=self.log).rev() {
            self.push(p >> i);
        }
        self.data[p] = x;
        for i in 1..=self.log {
            self.update(p >> i);
        }
    }

    pub fn get(&mut self, mut p: usize) -> <F::M as Monoid>::Target {
        assert!(p < self.range_size);
        p += self.leaf_size;
        for i in (1..=self.log).rev() {
            self.push(p >> i);
        }
        self.data[p].clone()
    }

    fn get_range<R: RangeBounds<usize>>(&self, range: R) -> (usize, usize) {
        let l = match range.start_bound() {
            Included(&l) => l,
            Excluded(&l) => l + 1,
            Unbounded => 0,
        };
        let r = match range.end_bound() {
            Included(&r) => r + 1,
            Excluded(&r) => r,
            Unbounded => self.range_size,
        };
        assert!(l <= r && r <= self.range_size);
        (l, r)
    }

    pub fn prod<R: RangeBounds<usize>>(&mut self, range: R) -> <F::M as Monoid>::Target {
        let (mut l, mut r) = self.get_range(range);
        if l == r {
            return F::M::id_element();
        }
        if l == 0 && r == self.range_size {
            return self.all_prod();
        }

        l += self.leaf_size;
        r += self.leaf_size;

        for i in (1..=self.log).rev() {
            if ((l >> i) << i) != l {
                self.push(l >> i);
            }
            if ((r >> i) << i) != r {
                self.push((r - 1) >> i);
            }
        }

        let mut sml = F::M::id_element();
        let mut smr = F::M::id_element();
        while l < r {
            if l & 1 != 0 {
                sml = F::M::binary_operation(&sml, &self.data[l]);
                l += 1;
            }
            if r & 1 != 0 {
                r -= 1;
                smr = F::M::binary_operation(&self.data[r], &smr);
            }
            l >>= 1;
            r >>= 1;
        }

        F::M::binary_operation(&sml, &smr)
    }

    pub fn all_prod(&self) -> <F::M as Monoid>::Target {
        self.data[1].clone()
    }

    pub fn apply(&mut self, mut p: usize, f: &F::A) {
        assert!(p < self.range_size);
        p += self.leaf_size;
        for i in (1..=self.log).rev() {
            self.push(p >> i);
        }
        f.apply(&mut self.data[p]);
        for i in 1..=self.log {
            self.update(p >> i);
        }
    }

    /// 作用の可換性を仮定しない上での区間適用  
    /// 可換な作用は`apply_range_commutative`の方が定数倍高速
    pub fn apply_range<R: RangeBounds<usize>>(&mut self, range: R, f: &F::A) {
        let (mut l, mut r) = self.get_range(range);
        if l == r {
            return;
        }

        l += self.leaf_size;
        r += self.leaf_size;

        // 非可換なので、先に上から伝播しておく
        for i in (1..=self.log).rev() {
            if ((l >> i) << i) != l {
                self.push(l >> i);
            }
            if ((r >> i) << i) != r {
                self.push((r - 1) >> i);
            }
        }

        {
            let l_copy = l;
            let r_copy = r;
            while l < r {
                if l & 1 != 0 {
                    self.all_apply(l, f);
                    l += 1;
                }
                if r & 1 != 0 {
                    r -= 1;
                    self.all_apply(r, f);
                }
                l >>= 1;
                r >>= 1;
            }
            l = l_copy;
            r = r_copy;
        }

        // 先に伝播しているのでlazyの値を考慮せずに更新できる
        for i in 1..=self.log {
            if ((l >> i) << i) != l {
                self.update(l >> i);
            }
            if ((r >> i) << i) != r {
                self.update((r - 1) >> i);
            }
        }
    }

    pub fn max_right<G>(&mut self, mut l: usize, g: G) -> usize
    where
        G: Fn(&<F::M as Monoid>::Target) -> bool,
    {
        assert!(l <= self.range_size);
        assert!(g(&F::M::id_element()));
        if l == self.range_size {
            return self.range_size;
        }
        l += self.leaf_size;
        for i in (1..=self.log).rev() {
            self.push(l >> i);
        }
        let mut sm = F::M::id_element();
        while {
            while l % 2 == 0 {
                l >>= 1;
            }
            if !g(&F::M::binary_operation(&sm, &self.data[l])) {
                while l < self.leaf_size {
                    self.push(l);
                    l *= 2;
                    let res = F::M::binary_operation(&sm, &self.data[l]);
                    if g(&res) {
                        sm = res;
                        l += 1;
                    }
                }
                return l - self.leaf_size;
            }
            sm = F::M::binary_operation(&sm, &self.data[l]);
            l += 1;
            {
                let l = l as isize;
                (l & -l) != l
            }
        } {}
        self.range_size
    }

    pub fn min_left<G>(&mut self, mut r: usize, g: G) -> usize
    where
        G: Fn(&<F::M as Monoid>::Target) -> bool,
    {
        assert!(r <= self.range_size);
        assert!(g(&F::M::id_element()));
        if r == 0 {
            return 0;
        }
        r += self.leaf_size;
        for i in (1..=self.log).rev() {
            self.push((r - 1) >> i);
        }
        let mut sm = F::M::id_element();
        while {
            r -= 1;
            while r > 1 && r % 2 != 0 {
                r >>= 1;
            }
            if !g(&F::M::binary_operation(&self.data[r], &sm)) {
                while r < self.leaf_size {
                    self.push(r);
                    r = 2 * r + 1;
                    let res = F::M::binary_operation(&self.data[r], &sm);
                    if g(&res) {
                        sm = res;
                        r -= 1;
                    }
                }
                return r + 1 - self.leaf_size;
            }
            sm = F::M::binary_operation(&self.data[r], &sm);
            {
                let r = r as isize;
                (r & -r) != r
            }
        } {}
        0
    }
}

impl<F: ActionMonoid> LazySegTree<F>
where
    F::A: Commutative,
{
    /// 可換な作用の区間適用
    pub fn apply_range_commutative<R: RangeBounds<usize>>(&mut self, range: R, f: &F::A) {
        let (mut l, mut r) = self.get_range(range);
        if l == r {
            return;
        }

        l += self.leaf_size;
        r += self.leaf_size;

        // 可換なのでここの伝播をサボる

        {
            let l_copy = l;
            let r_copy = r;
            while l < r {
                if l & 1 != 0 {
                    self.all_apply(l, f);
                    l += 1;
                }
                if r & 1 != 0 {
                    r -= 1;
                    self.all_apply(r, f);
                }
                l >>= 1;
                r >>= 1;
            }
            l = l_copy;
            r = r_copy;
        }

        // 伝播をサボったので、ここでlazyを考慮して更新する
        for i in 1..=self.log {
            if ((l >> i) << i) != l {
                self.update_considering_lazy(l >> i);
            }
            if ((r >> i) << i) != r {
                self.update_considering_lazy((r - 1) >> i);
            }
        }
    }

    fn update_considering_lazy(&mut self, k: usize) {
        self.data[k] = F::M::binary_operation(&self.data[2 * k], &self.data[2 * k + 1]);
        self.lazy[k].apply(&mut self.data[k]);
    }
}

impl<F: ActionMonoid> LazySegTree<F> {
    /// dataを子から更新する
    fn update(&mut self, k: usize) {
        self.data[k] = F::M::binary_operation(&self.data[2 * k], &self.data[2 * k + 1]);
    }
    /// 作用を適用し、lazyノードがあれば(子があれば)作用を合成する
    fn all_apply(&mut self, k: usize, f: &F::A) {
        f.apply(&mut self.data[k]);
        if k < self.leaf_size {
            self.lazy[k].composition(f);
        }
    }
    /// 作用を子に押し込む
    fn push(&mut self, k: usize) {
        let mut parent = F::A::id_action();
        std::mem::swap(&mut parent, &mut self.lazy[k]);
        self.all_apply(2 * k, &parent);
        self.all_apply(2 * k + 1, &parent);
    }
}

#[cfg(test)]
mod test {
    use super::*;
    use algebra::Action;
    use rand::prelude::*;

    #[test]
    fn test_max_right_min_left() {
        // 区間加算、区間和
        #[derive(Debug, Clone, Copy, PartialEq, Eq)]
        struct SumMonoid {
            sum: u64,
            len: u64,
        }
        impl Monoid for SumMonoid {
            type Target = Self;
            fn id_element() -> Self {
                Self { sum: 0, len: 0 }
            }
            fn binary_operation(a: &Self, b: &Self) -> Self {
                Self {
                    sum: a.sum + b.sum,
                    len: a.len + b.len,
                }
            }
        }

        #[derive(Debug, Clone, Copy, PartialEq, Eq)]
        struct AddAction(u64);
        impl Action for AddAction {
            type Target = SumMonoid;
            fn id_action() -> Self {
                Self(0)
            }
            fn composition(&mut self, rhs: &Self) {
                self.0 += rhs.0;
            }
            fn apply(&self, target: &mut Self::Target) {
                target.sum += self.0 * target.len;
            }
        }
        impl Commutative for AddAction {}

        struct RARRSQ;
        impl ActionMonoid for RARRSQ {
            type M = SumMonoid;
            type A = AddAction;
        }

        let mut rng = rand::thread_rng();
        let n = 1000;
        let mut seg = LazySegTree::<RARRSQ>::from(
            (0..n)
                .map(|_| SumMonoid {
                    sum: rng.gen_range(0..=20000),
                    len: 1,
                })
                .collect::<Vec<_>>(),
        );
        for _ in 0..n * 10 {
            let l = rng.gen_range(0..n);
            let r = rng.gen_range(l..n);
            let x = rng.gen_range(0..=20000);
            seg.apply_range_commutative(l..r, &AddAction(x));

            let l = rng.gen_range(0..n);
            let bound = rng.gen_range(1..=200_000);
            let max_right = seg.max_right(l, |x| x.sum < bound);
            assert!(seg.prod(l..max_right).sum < bound);
            assert!(max_right == n || seg.prod(l..max_right + 1).sum >= bound);

            let r = rng.gen_range(0..n);
            let bound = rng.gen_range(1..=2000_000);
            let min_left = seg.min_left(r, |x| x.sum < bound);
            assert!(seg.prod(min_left..r).sum < bound);
            assert!(min_left == 0 || seg.prod(min_left - 1..r).sum >= bound);
        }
    }
}
Traceback (most recent call last):
  File "/opt/hostedtoolcache/Python/3.13.9/x64/lib/python3.13/site-packages/onlinejudge_verify/documentation/build.py", line 71, in _render_source_code_stat
    bundled_code = language.bundle(stat.path, basedir=basedir, options={'include_paths': [basedir]}).decode()
                   ~~~~~~~~~~~~~~~^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "/opt/hostedtoolcache/Python/3.13.9/x64/lib/python3.13/site-packages/onlinejudge_verify/languages/rust.py", line 288, in bundle
    raise NotImplementedError
NotImplementedError
Back to top page