結果

問題 No.2163 LCA Sum Query
ユーザー noshi91noshi91
提出日時 2022-12-14 00:50:55
言語 C++17
(gcc 13.3.0 + boost 1.87.0)
結果
RE  
実行時間 -
コード長 24,101 bytes
コンパイル時間 2,079 ms
コンパイル使用メモリ 127,944 KB
実行使用メモリ 35,136 KB
最終ジャッジ日時 2024-11-07 18:04:16
合計ジャッジ時間 19,639 ms
ジャッジサーバーID
(参考情報)
judge4 / judge3
このコードへのチャレンジ
(要ログイン)

テストケース

テストケース表示
入力 結果 実行時間
実行使用メモリ
testcase_00 AC 2 ms
5,248 KB
testcase_01 AC 2 ms
5,248 KB
testcase_02 RE -
testcase_03 RE -
testcase_04 RE -
testcase_05 RE -
testcase_06 RE -
testcase_07 RE -
testcase_08 RE -
testcase_09 RE -
testcase_10 RE -
testcase_11 RE -
testcase_12 RE -
testcase_13 RE -
testcase_14 RE -
testcase_15 RE -
testcase_16 RE -
testcase_17 RE -
testcase_18 RE -
testcase_19 RE -
testcase_20 RE -
testcase_21 RE -
testcase_22 AC 618 ms
29,880 KB
testcase_23 RE -
testcase_24 AC 610 ms
29,884 KB
testcase_25 RE -
testcase_26 AC 512 ms
29,760 KB
testcase_27 RE -
testcase_28 AC 508 ms
29,796 KB
testcase_29 RE -
testcase_30 AC 366 ms
34,360 KB
testcase_31 RE -
testcase_32 AC 362 ms
33,084 KB
testcase_33 RE -
testcase_34 AC 294 ms
30,060 KB
testcase_35 AC 356 ms
29,792 KB
testcase_36 AC 286 ms
30,004 KB
testcase_37 AC 343 ms
30,104 KB
testcase_38 AC 396 ms
31,680 KB
testcase_39 RE -
testcase_40 AC 408 ms
31,316 KB
testcase_41 RE -
権限があれば一括ダウンロードができます

ソースコード

diff #

#include <array>
#include <cassert>
#include <utility>
#include <variant>
#include <vector>

/*

struct Info {
  using V;
  using E;
  using Point;
  using Path;
  static Point rake(Point, Point);
  static Point id();
  static Path to_path(V, Point);
  static Path compress(Path, E, Path);
  static Path reverse(Path);
  static Point to_point(E, Path);
};

*/

template <class Info> class top_tree {
  using V = typename Info::V;
  using E = typename Info::E;
  using Point = typename Info::Point;
  using Path = typename Info::Path;

  struct node_type;
  using node_ptr = node_type*;

  struct vertex_node {
    V v;
    Path path;

    vertex_node(V v_, Path path_) : v(std::move(v_)), path(std::move(path_)) {}
  };

  struct solid_edge_node {
    bool rev;
    E e;
    Path sum;

    solid_edge_node(E e_, Path sum_)
      : rev(false), e(std::move(e_)), sum(std::move(sum_)) {}
  };

  struct dashed_edge_node {
    E e;
    Point point;
    Point sum;

    dashed_edge_node(E e_, Point point_, Point sum_)
      : e(std::move(e_)), point(std::move(point_)), sum(std::move(sum_)) {}
  };

  using data_variant =
    std::variant<vertex_node, solid_edge_node, dashed_edge_node>;

  struct node_type {
    node_ptr p;
    std::array<node_ptr, 3> c;
    data_variant data;

    template <class... Args>
    node_type(Args &&... args)
      : p(nullptr), c({ nullptr, nullptr, nullptr }),
      data(std::forward<Args>(args)...) {}
  };

  static void link_child(node_type& par, const node_ptr ch, const int dir) {
    par.c[dir] = ch;
    if (ch) {
      ch->p = &par;
    }
  }

  static int get_dir(const node_type& node) {
    if (node.p) {
      for (int i = 0; i < 3; i++) {
        if (node.p->c[i] == &node) {
          return i;
        }
      }
      assert(false);
    }
    else {
      return 1;
    }
  }

  static void p_replace(node_type& prev, node_type& n) {
    if (prev.p) {
      prev.p->c[get_dir(prev)] = &n;
      n.p = prev.p;
    }
    else {
      n.p = nullptr;
    }
  }

  static Point get_sum_point(const node_ptr ptr) {
    if (ptr) {
      return std::get<dashed_edge_node>(ptr->data).sum;
    }
    else {
      return Info::id();
    }
  }

  static const Path& get_sum_path(const node_type& node) {
    struct {
      const Path& operator()(const vertex_node& v) const { return v.path; }
      const Path& operator()(const solid_edge_node& s) const { return s.sum; }
      const Path& operator()(const dashed_edge_node&) const {
        throw "top_tree: internal error";
      }
    } visitor{};
    return std::visit(visitor, node.data);
  }

  static void update(node_type& node) {
    struct {
      node_type& node;
      void operator()(vertex_node&) const { throw "top_tree: internal error"; }
      void operator()(solid_edge_node& s) const {
        s.sum = Info::compress(get_sum_path(*node.c[0]), s.e,
          get_sum_path(*node.c[2]));
      }
      void operator()(dashed_edge_node& d) const {
        d.sum = Info::rake(d.point, Info::rake(get_sum_point(node.c[0]),
          get_sum_point(node.c[2])));
      }
    } visitor{ node };
    std::visit(visitor, node.data);
  }

  static void rotate(node_type& node, const int dir) {
    node_type& ch = *node.c[dir ^ 2];
    ch.p = node.p;
    if (node.p) {
      node.p->c[get_dir(node)] = &ch;
    }
    link_child(node, ch.c[dir], dir ^ 2);
    update(node);
    link_child(ch, &node, dir);
  }

  static void splay(node_type& node) {
    while (true) {
      const int d0 = get_dir(node);
      if (d0 == 1) {
        break;
      }
      node_type& p = *node.p;
      const int d1 = get_dir(p);
      if (d1 == 1) {
        rotate(p, d0 ^ 2);
        break;
      }
      node_type& pp = *p.p;
      if (d0 == d1) {
        rotate(pp, d1 ^ 2);
        rotate(p, d0 ^ 2);
      }
      else {
        rotate(p, d0 ^ 2);
        rotate(pp, d1 ^ 2);
      }
    }
    update(node);
  }

  static void reverse(node_type& node) {
    struct {
      void operator()(vertex_node&) const {}
      void operator()(solid_edge_node& s) const {
        s.sum = Info::reverse(std::move(s.sum));
        s.rev = !s.rev;
      }
      void operator()(dashed_edge_node&) const {
        throw "top_tree: internal error";
      }
    } visitor{};
    std::visit(visitor, node.data);
  }

  static void propagate(node_type& node) {
    if (node.p) {
      propagate(*node.p);
    }
    struct {
      node_type& node;
      void operator()(vertex_node&) const {}
      void operator()(solid_edge_node& s) const {
        if (s.rev) {
          s.rev = false;
          std::swap(node.c[0], node.c[2]);
          reverse(*node.c[0]);
          reverse(*node.c[2]);
        }
      }
      void operator()(dashed_edge_node&) const {}
    } visitor{ node };
    std::visit(visitor, node.data);
  }

  static node_ptr merge(const node_ptr x, node_ptr y) {
    if (!y) {
      return x;
    }
    y->p = nullptr;
    while (y->c[0]) {
      y = y->c[0];
    }
    link_child(*y, x, 0);
    splay(*y);
    return y;
  }

  static void expose_edge(node_type& node) {
    propagate(node);
    node_ptr ptr = &node;
    if (ptr->data.index() == 1) {
      splay(*ptr);
      ptr = ptr->p;
    }
    while (ptr) {
      splay(*ptr);
      node_type& v = *ptr->p;
      if (get_dir(v) != 1) {
        splay(*v.p);
      }
      if (get_dir(v) != 1) {
        splay(*v.p);
      }
      if (get_dir(v) == 2 && get_dir(*v.p) == 0) {
        splay(*v.p);
      }
      if (get_dir(v) == 0) {
        node_type& p = *v.p;
        p_replace(p, *ptr);
        link_child(v, &p, 1);
        link_child(p, p.c[2], 1);
        link_child(p, ptr->c[0], 0);
        link_child(p, ptr->c[2], 2);
        E e = std::move(std::get<solid_edge_node>(p.data).e);
        Point point = Info::to_point(e, get_sum_path(*p.c[1]));
        Point sum = Info::rake(
          point, Info::rake(get_sum_point(p.c[0]), get_sum_point(p.c[2])));
        p.data = data_variant(std::in_place_type<dashed_edge_node>,
          std::move(e), std::move(point), std::move(sum));
      }
      else {
        link_child(v, merge(ptr->c[0], ptr->c[2]), 1);
        p_replace(v, *ptr);
      }
      vertex_node& inner = std::get<vertex_node>(v.data);
      inner.path = Info::to_path(inner.v, get_sum_point(v.c[1]));
      link_child(*ptr, &v, 0);
      link_child(*ptr, ptr->c[1], 2);
      ptr->c[1] = nullptr;
      E e = std::move(std::get<dashed_edge_node>(ptr->data).e);
      Path sum =
        Info::compress(get_sum_path(*ptr->c[0]), e, get_sum_path(*ptr->c[2]));
      ptr->data = data_variant(std::in_place_type<solid_edge_node>,
        std::move(e), std::move(sum));
      splay(*ptr);
      ptr = ptr->p;
    }
    splay(node);
  }

  static void check(const node_type& n) {
    if (n.p) {
      get_dir(n);
    }
    for (int i = 0; i < 3; i++) {
      if (n.c[i]) {
        assert(n.c[i]->p == &n);
      }
    }
    struct {
      const node_type& n;
      void operator()(const vertex_node&) const {
        assert(n.c[0] == nullptr);
        if (n.c[1]) {
          assert(n.c[1]->data.index() == 2);
        }
        assert(n.c[2] == nullptr);
        if (n.p) {
          assert(n.p->data.index() == 1 || n.p->data.index() == 2);
        }
      }
      void operator()(const solid_edge_node&) const {
        assert(n.c[0] != nullptr);
        assert(n.c[0]->data.index() == 0 || n.c[0]->data.index() == 1);
        assert(n.c[1] == nullptr);
        assert(n.c[2] != nullptr);
        assert(n.c[2]->data.index() == 0 || n.c[2]->data.index() == 1);
        if (n.p) {
          assert(n.p->data.index() == 1 || n.p->data.index() == 2);
        }
      }
      void operator()(const dashed_edge_node&) const {
        if (n.c[0]) {
          assert(n.c[0]->data.index() == 2);
        }
        assert(n.c[1] != nullptr);
        assert(n.c[1]->data.index() == 0 || n.c[1]->data.index() == 1);
        if (n.c[2]) {
          assert(n.c[2]->data.index() == 2);
        }
        assert(n.p != nullptr);
        assert(n.p->data.index() == 0 || n.p->data.index() == 2);
      }
    } visitor{ n };
    std::visit(visitor, n.data);
  }

  static void check_cp(node_type& n) {
    struct {
      node_type& n;
      void operator()(vertex_node&) const { check_ds(n.c[1]); }
      void operator()(solid_edge_node&) const {
        check_cp(*n.c[0]);
        check_cp(*n.c[2]);
      }
      void operator()(dashed_edge_node&) const { assert(false); }
    } visitor{ n };
    std::visit(visitor, n.data);
  }
  static void check_ds(const node_ptr ptr) {
    if (ptr) {
      struct {
        node_type& n;
        void operator()(vertex_node&) const { assert(false); }
        void operator()(solid_edge_node&) const { assert(false); }
        void operator()(dashed_edge_node&) const {
          check_ds(n.c[0]);
          check_cp(*n.c[1]);
          check_ds(n.c[2]);
        }
      } visitor{ *ptr };
      std::visit(visitor, ptr->data);
    }
  }

  std::vector<node_type> vertex_nodes;
  std::vector<node_type> edge_nodes;
  node_ptr free_list;

  template <class... Args> node_type& allocate(Args &&... args) {
    if (free_list) {
      node_type& n = *free_list;
      free_list = n.c[0];
      n = node_type(std::forward<Args>(args)...);
      return n;
    }
    else {
      edge_nodes.emplace_back(std::forward<Args>(args)...);
      return edge_nodes.back();
    }
  }

  node_type& expose_vertex(const int v_) {
    node_type& v = vertex_nodes[v_];
    if (v.p) {
      expose_edge(*v.p);
    }
    if (get_dir(v) == 2) {
      splay(*v.p);
    }
    if (get_dir(v) == 0) {
      node_type& p = *v.p;
      splay(p);
      p_replace(p, *p.c[0]);
      link_child(p, v.c[1], 0);
      link_child(p, p.c[2], 1);
      p.c[2] = nullptr;
      link_child(v, &p, 1);
      E e = std::move(std::get<solid_edge_node>(p.data).e);
      Point point = Info::to_point(e, get_sum_path(*p.c[1]));
      Point sum = Info::rake(
        point, Info::rake(get_sum_point(p.c[0]), get_sum_point(p.c[2])));
      p.data = data_variant(std::in_place_type<dashed_edge_node>, std::move(e),
        std::move(point), std::move(sum));
      vertex_node& inner = std::get<vertex_node>(v.data);
      inner.path = Info::to_path(inner.v, get_sum_point(v.c[1]));
    }
    if (v.p) {
      node_type& p = *v.p;
      splay(p);
      return p;
    }
    else {
      return v;
    }
  }

  node_type& evert(const int v) {
    node_type& r = expose_vertex(v);
    reverse(r);
    return r;
  }

public:
  class edge_handle {
    friend top_tree;
    node_ptr ptr;

    edge_handle(const node_ptr ptr_) : ptr(ptr_) {}

  public:
    edge_handle() : ptr(nullptr) {}
    E get() const {
      struct {
        E operator()(vertex_node&) const { throw "top_tree: internal error"; }
        E operator()(solid_edge_node& n) const { return n.e; }
        E operator()(dashed_edge_node& n) const { return n.e; }
      } visitor{};
      return std::visit(visitor, *ptr);
    }
  };

  top_tree(std::vector<V> vertices)
    : vertex_nodes(), edge_nodes(), free_list(nullptr) {
    const int n = vertices.size();
    vertex_nodes.reserve(n);
    for (int i = 0; i < n; i++) {
      Path sum = Info::to_path(vertices[i], Info::id());
      vertex_nodes.emplace_back(std::in_place_type<vertex_node>,
        std::move(vertices[i]), std::move(sum));
    }
    edge_nodes.reserve(n - 1);
  }

  void set_vertex(const int i, V v) {
    node_type& n = vertex_nodes[i];
    vertex_node& inner = std::get<vertex_node>(n.data);
    inner.v = std::move(v);
    inner.path = Info::to_path(inner.v, get_sum_point(n.c[1]));
    // check();
    if (n.p) {
      expose_edge(*n.p);
    }
    // check();
  }

  void set_edge(const edge_handle h, E e) {
    expose_edge(*h.ptr);
    std::get<solid_edge_node>(h.ptr->data).e = std::move(e);
    update(*h.ptr);
  }

  Path get_path(const int u, const int v) {
    evert(u);
    // check();
    node_type& r = expose_vertex(v);
    // check();
    return get_sum_path(r);
  }

  edge_handle link(const int u, const int v, E e) {
    node_type& u_ = expose_vertex(u);
    node_type& v_ = evert(v);
    Path sum = Info::compress(get_sum_path(u_), e, get_sum_path(v_));
    node_type& r = allocate(std::in_place_type<solid_edge_node>, std::move(e),
      std::move(sum));
    link_child(r, &u_, 0);
    link_child(r, &v_, 2);
    return edge_handle(&r);
  }

  void cut(const edge_handle h) {
    node_type& n = *h.ptr;
    expose_edge(n);
    n.c[0]->p = nullptr;
    n.c[2]->p = nullptr;
    n.c[0] = free_list;
    free_list = &n;
  }

  void check() {
    for (auto& v : vertex_nodes) {
      check(v);
    }
    std::vector<bool> used(edge_nodes.size(), true);
    {
      node_ptr p = free_list;
      while (p) {
        used[p - edge_nodes.data()] = false;
        p = p->c[0];
      }
    }
    for (int i = 0; i < edge_nodes.size(); i++) {
      if (used[i]) {
        check(edge_nodes[i]);
      }
    }

    for (auto& v : vertex_nodes) {
      if (!v.p) {
        check_cp(v);
      }
    }
    for (int i = 0; i < edge_nodes.size(); i++) {
      if (used[i] && !edge_nodes[i].p) {
        check_cp(edge_nodes[i]);
      }
    }
  }
};

#include <map>

using u64 = unsigned long long;

struct lcasum {
  struct V {
    int id;
    bool on;
  };
  struct E {};
  struct Point {
    u64 score;
    u64 hollow;
    int count;
  };

  struct Path {
    u64 scl, scr;
    u64 a;
    int count;
  };

  static Point rake(Point x, Point y) {
    return { x.score + y.score
     ,x.hollow + y.hollow + 
      u64(x.count) * u64(y.count), x.count + y.count };
  }
  static Point id() { return { 0, 0 }; }
  static Path to_path(V v, Point p) {
    if (v.on) {
      p.score += u64(v.id) * u64(p.count);
      p.count += 1;
    }
    p.score += p.hollow * u64(v.id);
    return { p.score, p.score, u64(p.count) * u64(v.id), p.count };
  }
  static Path compress(Path l, E, Path r) {
    return { l.scl + r.scl + l.a * r.count, l.scr + r.scr + r.a * l.count,
            l.a + r.a, l.count + r.count };
  }
  static Path reverse(Path p) {
    std::swap(p.scl, p.scr);
    return p;
  }
  static Point to_point(E, Path p) { return { p.scl, 0, p.count }; }
};

#include <algorithm>
#include <cassert>
#include <climits>
#include <cstdint>
#include <utility>
#include <vector>

namespace noshi91 {

  namespace tree_utility_impl {

    int bsr(int x) {
#if defined(__GNUC__)
      return 31 - __builtin_clz(x);
      #elif defined(_MSC_VER)
        unsigned long i;
      _BitScanReverse(&i, x);
      return i;
#else
      static_assert(false, "not implemented");
      return -1;
#endif
    }

    using std::vector;

    class tree_utility {
      class node_type {
      public:
        int subtree_size;
        int depth;
        int in;
        int out;
        int ladder;

        node_type() : subtree_size(1), depth(), in(), out(), ladder() {}
      };

      int n_;
      int root_;
      vector<node_type> nodes;
      vector<vector<int>> sparse_table;
      vector<vector<int>> jump_table;
      vector<int> ladder;

      void comparator_in(int& u, int& v) const {
        if (nodes[u].in > nodes[v].in) {
          std::swap(u, v);
        }
      }

      bool compare_by_depth(int u, int v) const {
        return nodes[u].depth < nodes[v].depth;
      }

      int min_by_depth(int u, int v) const {
        if (compare_by_depth(u, v)) {
          return u;
        }
        else {
          return v;
        }
      }

      int max_by_depth(int u, int v) const {
        if (compare_by_depth(u, v)) {
          return v;
        }
        else {
          return u;
        }
      }

      bool is_ancestor_of(int u, int v) const {
        return nodes[u].in <= nodes[v].in && nodes[v].in <= nodes[u].out;
      }

      int level_ancestor_0(int u, int d) const {
        d = nodes[u].depth - d;
        if (d == 0) {
          return u;
        }
        else {
          int p = bsr(d);
          d -= 1 << p;
          u = jump_table[p][u];
          return ladder[nodes[u].ladder - d];
        }
      }

      int lca_0_ordered(int u, int v) const {
        u = nodes[u].in;
        v = nodes[v].in + 1;
        int p = bsr(v - u);
        return min_by_depth(sparse_table[p][u], sparse_table[p][v - (1 << p)]);
      }

      int lca_0(int u, int v) const {
        comparator_in(u, v);
        return lca_0_ordered(u, v);
      }

      int distance_(int u, int v) const {
        int lca_ = lca_0(u, v);
        return nodes[u].depth + nodes[v].depth - 2 * nodes[lca_].depth;
      }

      int jump_(int u, int v, int d) const {
        if (d < 0) {
          return -1;
        }
        int lca_ = lca_0(u, v);
        if (nodes[u].depth - d >= nodes[lca_].depth) {
          return level_ancestor_0(u, nodes[u].depth - d);
        }
        else {
          int t = 2 * nodes[lca_].depth + d - nodes[u].depth;
          if (t <= nodes[v].depth) {
            return level_ancestor_0(v, t);
          }
          else {
            return -1;
          }
        }
      }

      int step_(int u, int v) const {
        if (is_ancestor_of(u, v)) {
          if (u == v) {
            return -1;
          }
          else {
            return level_ancestor_0(v, nodes[u].depth + 1);
          }
        }
        else {
          return jump_table[0][u];
        }
      }

      int meet_(int u, int v, int w) const {
        comparator_in(u, v);
        comparator_in(v, w);
        comparator_in(u, v);
        return max_by_depth(lca_0_ordered(u, v), lca_0_ordered(v, w));
      }

    public:
      explicit tree_utility(int n, const vector<std::pair<int, int>>& edges)
        : n_(n), root_(0), nodes(n), sparse_table(), jump_table(), ladder(2 * n) {
        assert(n > 0);
        vector<vector<int>> g(n);
        for (const auto& e : edges) {
          assert(0 <= e.first && e.first < n);
          assert(0 <= e.second && e.second < n);
          g[e.first].push_back(e.second);
          g[e.second].push_back(e.first);
        }

        vector<int> tour;
        tour.reserve(2 * n - 1);
        vector<int> pare(n, -1);
        {
          vector<int> height(n, 0);
          vector<int> used(n, 0);
          auto dfs = [&](auto& dfs, int v) -> void {
            used[v] = 1;
            nodes[v].in = tour.size();
            tour.push_back(v);
            for (int& u : g[v]) {
              assert(!used[u]);
              g[u].erase(std::find(g[u].begin(), g[u].end(), v));
              pare[u] = v;
              nodes[u].depth = nodes[v].depth + 1;
              dfs(dfs, u);
              nodes[v].subtree_size += nodes[u].subtree_size;
              if (height[v] <= height[u]) {
                height[v] = height[u] + 1;
                std::swap(g[v].front(), u);
              }
              tour.push_back(v);
            }
            nodes[v].out = tour.size();
          };
          nodes[0].depth = 0;
          dfs(dfs, 0);
          assert(
            std::all_of(used.begin(), used.end(), [](int x) { return x == 1; }));
        }

        {
          for (int w = 1; 2 * w < 2 * n; w *= 2) {
            int s = 2 * n - 2 * w;
            vector<int> next(s);
            for (int i = 0; i < s; ++i) {
              next[i] = min_by_depth(tour[i], tour[i + w]);
            }
            sparse_table.push_back(std::move(tour));
            tour = std::move(next);
          }
          sparse_table.push_back(std::move(tour));
        }

        {
          int times = bsr(n);
          for (int i = 0; i < times; ++i) {
            vector<int> next(n);
            for (int i = 0; i < n; ++i) {
              if (pare[i] == -1) {
                next[i] = -1;
              }
              else {
                next[i] = pare[pare[i]];
              }
            }
            jump_table.push_back(std::move(pare));
            pare = std::move(next);
          }
          jump_table.push_back(std::move(pare));
        }

        {
          int pos = 0;
          vector<int> path;
          path.reserve(n);
          auto dfs = [&](auto& dfs, int v, int d) -> void {
            path.push_back(v);
            bool pushed = false;
            for (int u : g[v]) {
              if (pushed) {
                dfs(dfs, u, path.size());
              }
              else {
                dfs(dfs, u, d);
                pushed = true;
              }
            }
            if (!pushed) {
              int p = path.size();
              int s = std::max(0, d - (p - d));
              std::copy(path.begin() + s, path.end(), ladder.begin() + pos);
              for (int i = d; i < p; ++i) {
                nodes[path[i]].ladder = pos + (d - s) + i;
              }
              pos += p - s;
            }
            path.pop_back();
          };
          dfs(dfs, 0, 0);
        }
      }

      int size() const { return n_; }

      int root() const { return root_; }

      void reroot(int new_root) { root_ = new_root; }

      int depth(int v) const {
        assert(0 <= v && v < n_);
        return distance_(root_, v);
      }

      int subtree_size(int v) const {
        if (is_ancestor_of(root_, v)) {
          if (root_ == v) {
            return n_;
          }
          else {
            return nodes[v].subtree_size;
          }
        }
        else {
          int c = level_ancestor_0(root_, nodes[v].depth + 1);
          return n_ - nodes[c].subtree_size;
        }
      }

      int level_ancestor(int v, int d) const {
        assert(0 <= v && v < n_);
        return jump_(root_, v, d);
      }

      int kth_ancestor(int v, int k) const {
        assert(0 <= v && v < n_);
        return jump_(v, root_, k);
      }

      int parent(int v) const {
        assert(0 <= v && v < n_);
        return step_(v, root_);
      }

      int lca(int u, int v) const {
        assert(0 <= u && u < n_);
        assert(0 <= v && v < n_);
        return meet_(root_, u, v);
      }

      int distance(int u, int v) const {
        assert(0 <= u && u < n_);
        assert(0 <= v && v < n_);
        return distance_(u, v);
      }

      int cut_size(int u, int v) const {
        assert(0 <= u && u < n_);
        assert(0 <= v && v < n_);
        if (jump_table[0][u] == v) {
          return nodes[u].subtree_size;
        }
        else if (jump_table[0][v] == u) {
          return n_ - nodes[v].subtree_size;
        }
        else {
          return -1;
        }
      }

      int jump(int u, int v, int d) const {
        assert(0 <= u && u < n_);
        assert(0 <= v && v < n_);
        return jump_(u, v, d);
      }

      int step(int u, int v) const {
        assert(0 <= u && u < n_);
        assert(0 <= v && v < n_);
        return step_(u, v);
      }

      int meet(int u, int v, int w) const {
        assert(0 <= u && u < n_);
        assert(0 <= v && v < n_);
        assert(0 <= w && w < n_);
        return meet_(u, v, w);
      }
    };

  } // namespace tree_utility_impl

  using tree_utility_impl::tree_utility;

} // namespace noshi91

#include <iostream>
#include <utility>
#include <vector>

int main() {
  std::ios::sync_with_stdio(false);
  std::cin.tie(nullptr);

  int N, Q;
  std::cin >> N >> Q;
  std::vector<lcasum::V> vs(N);
  for (int i = 0; i < N; i++) {
    vs[i] = { i+1, false };
  }
  top_tree<lcasum> tree(vs);

  std::map<std::pair<int, int>, typename top_tree<lcasum>::edge_handle> map;

  std::vector<std::pair<int, int>> es;

  for (int i = 0; i < N - 1; i++) {
    int a, b;
    std::cin >> a >> b;
    a -= 1;
    b -= 1;
    map[std::minmax(a, b)] = tree.link(a, b, {});
    es.push_back({ a, b });
  }

  const noshi91::tree_utility ut(N, es);

  for (int i = 0; i < Q; i++) {
    int u, r, v;
    std::cin >> u >> r >> v;
    u--;
    r--;
    v--;
    vs[u].on ^= 1;
    tree.set_vertex(u, vs[u]);

    if (r == v) {
      std::cout << tree.get_path(r, r).scl << "\n";
    }
    else {
      int p = ut.step(v, r);
      //std::cerr << "## " << v << " " << p << " " << r << "\n";
      auto& ref = map[std::minmax(v, p)];
      tree.cut(ref);
      std::cout << tree.get_path(v, v).scl << "\n";
      ref = tree.link(v, p, {});
    }
  }

  return 0;
}
0