imsuck's library

This documentation is automatically generated by competitive-verifier/competitive-verifier

View the Project on GitHub imsuck/library

:heavy_check_mark: test/yosupo/tree/dynamic_tree_vertex_set_path_composite_top_tree_ecnerwala.test.cpp

Depends on

Code

#define PROBLEM "https://judge.yosupo.jp/problem/dynamic_tree_vertex_set_path_composite"

#include <bits/stdc++.h>
using namespace std;

#include "math/modint.hpp"
#include "other/types.hpp"
#include "tree/top_tree_ecnerwala.hpp"

using mint = modint998;

struct affine {
    mint a = 1, b = 0;
    affine operator()(const affine &g) { return {a * g.a, a * g.b + b}; }
    mint operator()(mint x) { return a * x + b; }
};

struct node : top_tree_node_base<node> {
    affine val;
    array<affine, 2> f;

    void flip() { swap(f[0], f[1]); }

    void pull() {
        f.fill(is_vert ? val : affine{1, 0});
        if (c[0]) {
            f[0] = f[0](c[0]->f[0]);
            f[1] = c[0]->f[1](f[1]);
        }
        if (c[1]) {
            f[0] = c[1]->f[0](f[0]);
            f[1] = f[1](c[1]->f[1]);
        }
    }
};

int main() {
    cin.tie(nullptr)->sync_with_stdio(false);
    int n, q;
    cin >> n >> q;

    vector<node> a(n);
    for (auto &x : a) {
        cin >> x.val.a >> x.val.b;
        x.set_vert();
    }

    map<pair<i32, i32>, node> edges;
    auto es = [&](i32 u, i32 v) { return &edges[minmax(u, v)]; };

    for (i32 i = 0; i < n - 1; i++) {
        i32 u, v;
        cin >> u >> v;
        link(es(u, v), &a[u], &a[v]);
    }

    while (q--) {
        i32 op;
        cin >> op;
        if (op == 0) {
            i32 u, v, w, x;
            cin >> u >> v >> w >> x;
            auto it = edges.find(minmax(u, v));
            cut(&it->second);
            edges.erase(it);
            link(es(w, x), &a[w], &a[x]);
        } else if (op == 1) {
            i32 p, c, d;
            cin >> p >> c >> d;
            a[p].expose();
            a[p].val = {c, d};
            a[p].pull_all();
        } else {
            i32 u, v, x;
            cin >> u >> v >> x;
            a[u].evert();
            affine f = get_path(&a[v])->f[0];
            cout << f(x) << "\n";
        }
    }
}
#line 1 "test/yosupo/tree/dynamic_tree_vertex_set_path_composite_top_tree_ecnerwala.test.cpp"
#define PROBLEM "https://judge.yosupo.jp/problem/dynamic_tree_vertex_set_path_composite"

#include <bits/stdc++.h>
using namespace std;

#line 2 "math/modint.hpp"

// clang-format off
template<uint32_t m> struct modint {
    static_assert(m >= 1, "Modulus must be in the range [1;2^31)");

    using mint = modint;
    static constexpr bool is_simple = true;

    static constexpr uint32_t mod() noexcept { return m; }
    constexpr modint() noexcept = default;
    constexpr modint(int64_t v) noexcept : _v(uint32_t((v %= m) < 0 ? v + m : v)) {}
    constexpr static mint raw(uint32_t v) noexcept { mint x; return x._v = v, x; }
    template<class T> constexpr explicit operator T() const noexcept { return _v; }

    constexpr mint &operator++() noexcept { return _v = ++_v == mod() ? 0 : _v, *this; }
    constexpr mint &operator--() noexcept { --(_v ? _v : _v = mod()); return *this; }
    constexpr mint operator++(int) noexcept { return exchange(*this, ++mint(*this)); }
    constexpr mint operator--(int) noexcept { return exchange(*this, --mint(*this)); }

    constexpr mint &operator+=(mint rhs) noexcept {
        return _v = int(_v += rhs._v - mod()) < 0 ? _v + mod() : _v, *this;
    }
    constexpr mint &operator-=(mint rhs) noexcept {
        return _v = int(_v -= rhs._v) < 0 ? _v + mod() : _v, *this;
    }
    constexpr mint &operator*=(mint rhs) noexcept {
        return _v = uint64_t(_v) * rhs._v % mod(), *this;
    }
    constexpr mint &operator/=(mint rhs) noexcept {
        return *this = *this * rhs.inv();
    }

    constexpr friend mint operator+(mint l, mint r) noexcept { return l += r; }
    constexpr friend mint operator-(mint l, mint r) noexcept { return l -= r; }
    constexpr friend mint operator*(mint l, mint r) noexcept { return l *= r; }
    constexpr friend mint operator/(mint l, mint r) noexcept { return l /= r; }

    constexpr mint operator+() const noexcept { return *this; }
    constexpr mint operator-() const noexcept { return raw(_v ? mod() - _v : 0); }

    constexpr friend bool operator==(mint l, mint r) noexcept { return l._v == r._v; }
    constexpr friend bool operator!=(mint l, mint r) noexcept { return l._v != r._v; }
    constexpr friend bool operator<(mint l, mint r) noexcept { return l._v < r._v; }

    constexpr mint pow(uint64_t n) const noexcept {
        mint b = *this, res = 1;
        while (n) n & 1 ? res *= b : 0, b *= b, n >>= 1;
        return res;
    }

    constexpr mint inv() const noexcept {
        int a = _v, b = mod(), x = 1, y = 0;
        while (b) {
            x = exchange(y, x - a / b * y);
            a = exchange(b, a % b);
        }
        assert(a == 1);
        return x;
    }

    friend istream &operator>>(istream &is, mint &x) {
        int64_t v{};
        return is >> v, x = v, is;
    }
    friend ostream &operator<<(ostream &os, const mint &x) { return os << x._v; }

  private:
    uint32_t _v = 0;
};
using modint107 = modint<1'000'000'007>;
using modint998 = modint<998'244'353>;
// clang-format on
#line 2 "other/types.hpp"

using i8 = int8_t;
using u8 = uint8_t;
using i32 = int32_t;
using u32 = uint32_t;
using i64 = int64_t;
using u64 = uint64_t;
using f32 = float;
using f64 = double;
using f80 = long double;
using str = string;
template<class T> using vec = vector<T>;
template<class E = i32> using graph = vec<vec<E>>;
#line 2 "tree/top_tree_ecnerwala.hpp"

/**
 * https://github.com/ecnerwala/cp-book/blob/master/src/top_tree.hpp
 * Convenient memo:
 *   Creating vertices:
 *     n->val = ...;
 *     n->set_vert();
 *
 *   Creating edges:
 *     link(e, va, vb);
 *
 *   Updates:
 *     auto cur = get_path(va, vb); // or get_subtree(va, vb)
 *     cur->do_stuff();
 *     cur->pull_all();
 *
 * Node types:
 *   path edges: compress(c[0], self, c[1])
 *     assert(is_path && !is_vert);
 *     assert(c[0] && c[1]);
 *     assert(c[0]->is_path && c[1]->is_path);
 *     assert(!c[2]);
 *   (path) vertices: add_vertex(self, rake(c[0], c[1]))
 *     assert(is_path && is_vert);
 *     assert(!c[2]);
 *     if (c[0]) assert(!c[0]->is_path);
 *     if (c[1]) assert(!c[1]->is_path);
 *   non-path edges: rake(c[0], add_edge(self, c[2]), c[1])
 *     assert(!is_path && !is_vert);
 *     assert(c[2])
 *     assert(c[2]->is_path);
 *     if (c[0]) assert(!c[0]->is_path);
 *     if (c[1]) assert(!c[1]->is_path);
 */
template<class node> struct top_tree_node_base {
    using ptr = node *;

  private:
    ptr as_derived() { return (ptr)this; }
    auto as_derived() const { return (const ptr)this; }

  public:
    bool rev = false;
    ptr p = 0;
    array<ptr, 3> c{0, 0, 0};

    int d() const {
        assert(p);
        if (this == p->c[0]) return 0;
        if (this == p->c[1]) return 1;
        if (this == p->c[2]) return 2;
        assert(false);
    }
    ptr &p_c() { return p->c[d()]; }

    bool is_vert, is_path;
    void set_vert() { is_path = is_vert = true, _pull(); }

    bool r() const { return !p || p->is_path != is_path; }

  protected:
#define CHECK(a) \
    template<class T, class = void> struct has_##a##_t : false_type {}; \
    template<class T> struct has_##a##_t<T, void_t<decltype(&T::a)>> : true_type {}; \
    template<class T> static constexpr bool has_##a = has_##a##_t<T>::value;
    CHECK(push) CHECK(flip) CHECK(pull)
#undef CHECK
    void _flip() {
        assert(is_path);
        swap(c[0], c[1]), rev ^= 1;
        if constexpr (has_flip<node>) as_derived()->flip();
    }
    void push_rev() {
        if (rev) {
            assert(is_path);
            if (!is_vert) {
                c[0]->_flip();
                c[1]->_flip();
            }
            rev = false;
        }
    }
    void _push() {
        push_rev();
        if constexpr (has_push<node>) as_derived()->push();
    }
    void _pull() {
        if constexpr (has_pull<node>) as_derived()->pull();
    }

  public:
    void push_all() {
        if (p) p->push_all();
        _push();
    }
    void push_flip_all() {
        if (p) p->push_flip_all();
        if (rev) {
            assert(is_path);
            if (!is_vert) {
                c[0]->_flip();
                c[1]->_flip();
            }
            rev = false;
        }
    }
    ptr pull_all() {
        ptr cur = as_derived();
        cur->_push(), cur->_pull();
        for (; cur->p; cur->_pull()) cur = cur->p;
        return cur;
    }

  private:
    static inline void attach(ptr pa, int c_d, ptr ch) {
        pa->c[c_d] = ch;
        if (ch) ch->p = pa;
    }
    void rot() {
        assert(!is_vert);
        assert(!r());

        ptr pa = p;
        int x = d();
        assert(x != 2);
        ptr ch = c[!x];

        if (pa->p) pa->p_c() = as_derived();
        this->p = pa->p;

        attach(pa, x, ch);
        attach(as_derived(), !x, pa);

        pa->_pull();
    }
    void rot2(int c_d) {
        assert(!is_vert);
        assert(!r());
        assert(c[c_d]);
        assert(!c[c_d]->is_vert);

        if (d() == c_d) return rot();

        ptr pa = p;
        int x = d();
        assert(x != 2);
        ptr ch = c[c_d]->c[!x];

        if (pa->p) pa->p_c() = as_derived();
        this->p = pa->p;

        attach(pa, x, ch);
        attach(c[c_d], !x, pa);

        pa->_pull();
    }

    void splay_dir(int x) {
        for (; !r() && d() == x; rot())
            if (!p->r() && p->d() == x) p->rot();
    }

    void splay() {
        assert(!is_vert);
        for (; !r(); rot())
            if (!p->r()) p->d() == d() ? p->rot() : rot();
    }

    void splay2(int c_d) {
        assert(!is_vert && is_path);
        assert(c[c_d] && !c[c_d]->is_vert);
        for (; !r(); rot2(c_d))
            if (!p->r()) p->d() == d() ? p->rot() : rot2(c_d);
    }

    void splay2() {
        assert(!is_vert && is_path);
        assert(!r());
        p->splay2(d());
    }

    void splay_vert() {
        assert(is_vert);

        if (r()) return;

        p->splay_dir(d());
        if (p->r()) return;
        assert(p->d() != d());

        if (d() == 1) p->rot();
        assert(d() == 0);

        p->splay2();
        assert(d() == 0);
        assert(p->d() == 1);
        assert(p->p->r());
    }

    ptr cut_right() {
        assert(is_vert && is_path);
        splay_vert();

        if (r() || d() == 1) {
            assert(r() || (d() == 1 && p->r()));
            assert(!c[0]);
            return 0;
        }

        ptr pa = p;
        assert(pa->r() || (pa->d() == 1 && pa->p->r()));
        assert(!pa->is_vert);
        assert(pa->is_path);
        assert(pa->c[0] == this);
        assert(!pa->c[2]);

        if (pa->p) pa->p_c() = as_derived();
        this->p = pa->p;

        pa->is_path = false;
        pa->c[2] = pa->c[1];

        attach(pa, 0, c[0]);
        attach(pa, 1, c[1]);

        c[0] = 0;
        attach(as_derived(), 1, pa);
        assert(!c[2]);

        return pa->_pull(), pa;
    }

    ptr splice_non_path() {
        assert(!is_vert);
        assert(!is_path);

        splay();
        assert(p && p->is_vert && p->is_path);
        p->cut_right();

        if (!p->is_path) rot();
        assert(p && p->is_vert && p->is_path);
        assert(p->r() || (p->d() == 1 && p->p->r()));
        assert(p->c[d()] == this && !p->c[!d()]);

        ptr pa = p;

        if (pa->p) pa->p_c() = as_derived();
        this->p = pa->p;

        attach(pa, 0, c[0]);
        attach(pa, 1, c[1]);

        assert(c[2] && c[2]->is_path);
        attach(as_derived(), 0, pa);
        c[1] = c[2], c[2] = 0;

        is_path = true;

        return pa->_pull(), pa;
    }

    ptr splice_all() {
        ptr res = as_derived();
        for (ptr cur = as_derived(); cur; cur = cur->p) {
            if (!cur->is_path) res = cur->splice_non_path();
            assert(cur->is_path);
        }
        return res;
    }

  public:
    ptr expose() {
        assert(is_vert);

        push_all();

        ptr res = splice_all();
        cut_right(), pull_all();

        return res;
    }

    ptr expose_edge() {
        assert(!is_vert);

        push_all();

        ptr v = is_path ? c[1] : c[2];
        for (v->_push(); !v->is_vert; v->_push()) v = v->c[0];

        ptr res = v->splice_all();
        v->cut_right(), v->pull_all();

        assert(!p);
        assert(v == c[1]);

        return res == v ? as_derived() : res;
    }

    ptr meld_path_end() {
        assert(!p);
        ptr rt = as_derived();
        while (true) {
            rt->_push();
            if (rt->is_vert) break;
            rt = rt->c[1];
        }
        rt->splay_vert();
        if (rt->c[0] && rt->c[1]) {
            ptr ch = rt->c[1];
            while (true) {
                ch->_push();
                if (!ch->c[0]) break;
                ch = ch->c[0];
            }
            ch->splay();

            attach(ch, 0, rt->c[0]), rt->c[0] = 0;

            ch->_pull();
        } else if (rt->c[0]) {
            rt->c[1] = rt->c[0], rt->c[0] = 0;
        }
        return rt->pull_all();
    }

    void evert() {
        expose();

        ptr rt = as_derived();
        while (rt->p) {
            assert(rt->d() == 1);
            rt = rt->p;
        }

        rt->_flip();
        rt->meld_path_end();

        expose();

        assert(!p);
    }

    friend void link(ptr e, ptr v, ptr p) {
        assert(e && v && p);
        assert(!e->c[0] && !e->c[1] && !e->c[2]);
        v->evert(), p->expose();
        while (p->p) p = p->p;

        assert(!v->p);

        e->is_path = true, e->is_vert = false;
        attach(e, 0, p);
        attach(e, 1, v);
        e->_pull();
    }
    friend void link(node &e, node &v, node &p) { link(&e, &v, &p); }

    // Link v's root as a child of p with edge e
    // Returns false if they're already in the same subtree
    friend bool link_root(ptr e, ptr v, ptr p) {
        assert(e && v && p);
        assert(!e->c[0] && !e->c[1] && !e->c[2]);
        v->expose();
        p->expose();

        while (v->p) v = v->p;
        while (p->p) p = p->p;
        if (v == p) return false;

        e->is_path = true, e->is_vert = false;
        attach(e, 0, p);
        attach(e, 1, v);
        e->_pull();

        return true;
    }
    friend bool link_root(node &e, node &v, node &p) {
        return link_root(&e, &v, &p);
    }

    friend pair<ptr, ptr> cut(ptr e) {
        assert(!e->is_vert);
        e->expose_edge();

        assert(!e->p);
        assert(e->is_path);

        ptr l = e->c[0], r = e->c[1];
        assert(l && r);
        assert(!e->c[2]);

        e->c[0] = e->c[1] = 0;
        l->p = r->p = 0;

        l->meld_path_end();

        return {l, r};
    }
    friend pair<node &, node &> cut(node &e) {
        auto r = cut(&e);
        return {*r.first, *r.second};
    }

    friend ptr get_path(ptr v) {
        assert(v->is_vert);
        v->expose();
        if (!v->p) return v;
        assert(!v->p->p);
        return v->p;
    }
    friend node &get_path(node &v) { return *get_path(&v); }

    friend ptr get_subtree(ptr v) { return v->expose(), v; }
    friend node &get_subtree(node &v) { return v.expose(), v; }

    friend ptr lca(ptr u, ptr v) {
        if (u == v) return u;
        u->expose();
        auto up = u->p;
        ptr l = v->expose();
        if (u != v && up == u->p && (!up || !up->p)) return 0;
        return l;
    }
    friend ptr lca(node &u, node &v) { return lca(&u, &v); }

    template<class F> friend node *search(node *p, F &&select) {
        while (node *nxt = select(p)) {
            assert(p->is_vert);
            while (true) {
                p->_push(), nxt = select(p);
                if (nxt == p->c[2]) break;
                p = nxt;
            }
            p = nxt;
            assert(p->is_path);
            while (!p->is_vert) p->_push(), p = select(p);
        }
        p->evert();
        assert(!select(p));
        return p;
    }
    template<class F> friend node *search(node &p, F &&select) {
        return search(&p, forward<F>(select));
    }
};
#line 9 "test/yosupo/tree/dynamic_tree_vertex_set_path_composite_top_tree_ecnerwala.test.cpp"

using mint = modint998;

struct affine {
    mint a = 1, b = 0;
    affine operator()(const affine &g) { return {a * g.a, a * g.b + b}; }
    mint operator()(mint x) { return a * x + b; }
};

struct node : top_tree_node_base<node> {
    affine val;
    array<affine, 2> f;

    void flip() { swap(f[0], f[1]); }

    void pull() {
        f.fill(is_vert ? val : affine{1, 0});
        if (c[0]) {
            f[0] = f[0](c[0]->f[0]);
            f[1] = c[0]->f[1](f[1]);
        }
        if (c[1]) {
            f[0] = c[1]->f[0](f[0]);
            f[1] = f[1](c[1]->f[1]);
        }
    }
};

int main() {
    cin.tie(nullptr)->sync_with_stdio(false);
    int n, q;
    cin >> n >> q;

    vector<node> a(n);
    for (auto &x : a) {
        cin >> x.val.a >> x.val.b;
        x.set_vert();
    }

    map<pair<i32, i32>, node> edges;
    auto es = [&](i32 u, i32 v) { return &edges[minmax(u, v)]; };

    for (i32 i = 0; i < n - 1; i++) {
        i32 u, v;
        cin >> u >> v;
        link(es(u, v), &a[u], &a[v]);
    }

    while (q--) {
        i32 op;
        cin >> op;
        if (op == 0) {
            i32 u, v, w, x;
            cin >> u >> v >> w >> x;
            auto it = edges.find(minmax(u, v));
            cut(&it->second);
            edges.erase(it);
            link(es(w, x), &a[w], &a[x]);
        } else if (op == 1) {
            i32 p, c, d;
            cin >> p >> c >> d;
            a[p].expose();
            a[p].val = {c, d};
            a[p].pull_all();
        } else {
            i32 u, v, x;
            cin >> u >> v >> x;
            a[u].evert();
            affine f = get_path(&a[v])->f[0];
            cout << f(x) << "\n";
        }
    }
}

Test cases

Env Name Status Elapsed Memory
g++ almost_line_00 :heavy_check_mark: AC 833 ms 42 MB
g++ almost_line_01 :heavy_check_mark: AC 836 ms 42 MB
g++ example_00 :heavy_check_mark: AC 5 ms 4 MB
g++ example_01 :heavy_check_mark: AC 4 ms 4 MB
g++ issue_1216_00 :heavy_check_mark: AC 216 ms 47 MB
g++ max_random_00 :heavy_check_mark: AC 997 ms 42 MB
g++ max_random_01 :heavy_check_mark: AC 1039 ms 42 MB
g++ max_random_02 :heavy_check_mark: AC 1019 ms 42 MB
g++ medium_00 :heavy_check_mark: AC 6 ms 4 MB
g++ medium_01 :heavy_check_mark: AC 5 ms 4 MB
g++ medium_02 :heavy_check_mark: AC 5 ms 4 MB
g++ medium_03 :heavy_check_mark: AC 5 ms 4 MB
g++ medium_04 :heavy_check_mark: AC 6 ms 4 MB
g++ random_00 :heavy_check_mark: AC 629 ms 28 MB
g++ random_01 :heavy_check_mark: AC 847 ms 33 MB
g++ random_02 :heavy_check_mark: AC 386 ms 14 MB
g++ random_03 :heavy_check_mark: AC 451 ms 36 MB
g++ random_04 :heavy_check_mark: AC 212 ms 6 MB
g++ small_00 :heavy_check_mark: AC 5 ms 4 MB
g++ small_01 :heavy_check_mark: AC 4 ms 3 MB
g++ small_02 :heavy_check_mark: AC 4 ms 4 MB
g++ small_03 :heavy_check_mark: AC 4 ms 4 MB
g++ small_04 :heavy_check_mark: AC 4 ms 4 MB
clang++ almost_line_00 :heavy_check_mark: AC 862 ms 42 MB
clang++ almost_line_01 :heavy_check_mark: AC 860 ms 42 MB
clang++ example_00 :heavy_check_mark: AC 5 ms 4 MB
clang++ example_01 :heavy_check_mark: AC 4 ms 4 MB
clang++ issue_1216_00 :heavy_check_mark: AC 226 ms 49 MB
clang++ max_random_00 :heavy_check_mark: AC 1073 ms 42 MB
clang++ max_random_01 :heavy_check_mark: AC 1098 ms 42 MB
clang++ max_random_02 :heavy_check_mark: AC 1098 ms 42 MB
clang++ medium_00 :heavy_check_mark: AC 6 ms 4 MB
clang++ medium_01 :heavy_check_mark: AC 5 ms 4 MB
clang++ medium_02 :heavy_check_mark: AC 5 ms 4 MB
clang++ medium_03 :heavy_check_mark: AC 5 ms 4 MB
clang++ medium_04 :heavy_check_mark: AC 6 ms 4 MB
clang++ random_00 :heavy_check_mark: AC 710 ms 28 MB
clang++ random_01 :heavy_check_mark: AC 881 ms 33 MB
clang++ random_02 :heavy_check_mark: AC 389 ms 14 MB
clang++ random_03 :heavy_check_mark: AC 512 ms 36 MB
clang++ random_04 :heavy_check_mark: AC 228 ms 6 MB
clang++ small_00 :heavy_check_mark: AC 4 ms 4 MB
clang++ small_01 :heavy_check_mark: AC 4 ms 4 MB
clang++ small_02 :heavy_check_mark: AC 4 ms 4 MB
clang++ small_03 :heavy_check_mark: AC 4 ms 4 MB
clang++ small_04 :heavy_check_mark: AC 4 ms 4 MB
Back to top page