結果
問題 | No.235 めぐるはめぐる (5) |
ユーザー |
|
提出日時 | 2020-09-16 16:10:54 |
言語 | C++17 (gcc 13.3.0 + boost 1.87.0) |
結果 |
AC
|
実行時間 | 1,012 ms / 10,000 ms |
コード長 | 13,374 bytes |
コンパイル時間 | 2,848 ms |
コンパイル使用メモリ | 229,832 KB |
最終ジャッジ日時 | 2025-01-14 15:15:02 |
ジャッジサーバーID (参考情報) |
judge5 / judge3 |
(要ログイン)
ファイルパターン | 結果 |
---|---|
other | AC * 3 |
ソースコード
#include <bits/stdc++.h>#ifdef DEBUG#include <Mylib/Debug/debug.cpp>#else#define dump(...) ((void)0)#endiftemplate <typename T, typename U>bool chmin(T &a, const U &b){return (a > b ? a = b, true : false);}template <typename T, typename U>bool chmax(T &a, const U &b){return (a < b ? a = b, true : false);}template <typename T, size_t N, typename U>void fill_array(T (&a)[N], const U &v){std::fill((U*)a, (U*)(a + N), v);}template <typename T, size_t N, size_t I = N>auto make_vector(const std::array<int, N> &a, T value = T()){static_assert(I >= 1);static_assert(N >= 1);if constexpr (I == 1){return std::vector<T>(a[N - I], value);}else{return std::vector(a[N - I], make_vector<T, N, I - 1>(a, value));}}template <typename T>std::ostream& operator<<(std::ostream &s, const std::vector<T> &a){for(auto it = a.begin(); it != a.end(); ++it){if(it != a.begin()) s << " ";s << *it;}return s;}template <typename T>std::istream& operator>>(std::istream &s, std::vector<T> &a){for(auto &x : a) s >> x;return s;}/*** @title Modint* @docs mint.md*/namespace haar_lib {template <int32_t M>class modint {constexpr static int32_t MOD = M;uint32_t val;public:constexpr static auto mod(){return MOD;}constexpr modint(): val(0){}constexpr modint(int64_t n){if(n >= M) val = n % M;else if(n < 0) val = n % M + M;else val = n;}constexpr auto& operator=(const modint &a){val = a.val; return *this;}constexpr auto& operator+=(const modint &a){if(val + a.val >= M) val = (uint64_t)val + a.val - M;else val += a.val;return *this;}constexpr auto& operator-=(const modint &a){if(val < a.val) val += M;val -= a.val;return *this;}constexpr auto& operator*=(const modint &a){val = (uint64_t)val * a.val % M;return *this;}constexpr auto& operator/=(const modint &a){val = (uint64_t)val * a.inv().val % M;return *this;}constexpr auto operator+(const modint &a) const {return modint(*this) += a;}constexpr auto operator-(const modint &a) const {return modint(*this) -= a;}constexpr auto operator*(const modint &a) const {return modint(*this) *= a;}constexpr auto operator/(const modint &a) const {return modint(*this) /= a;}constexpr bool operator==(const modint &a) const {return val == a.val;}constexpr bool operator!=(const modint &a) const {return val != a.val;}constexpr auto& operator++(){*this += 1; return *this;}constexpr auto& operator--(){*this -= 1; return *this;}constexpr auto operator++(int){auto t = *this; *this += 1; return t;}constexpr auto operator--(int){auto t = *this; *this -= 1; return t;}constexpr static modint pow(int64_t n, int64_t p){if(p < 0) return pow(n, -p).inv();int64_t ret = 1, e = n % M;for(; p; (e *= e) %= M, p >>= 1) if(p & 1) (ret *= e) %= M;return ret;}constexpr static modint inv(int64_t a){int64_t b = M, u = 1, v = 0;while(b){int64_t t = a / b;a -= t * b; std::swap(a, b);u -= t * v; std::swap(u, v);}u %= M;if(u < 0) u += M;return u;}constexpr static auto frac(int64_t a, int64_t b){return modint(a) / modint(b);}constexpr auto pow(int64_t p) const {return pow(val, p);}constexpr auto inv() const {return inv(val);}friend constexpr auto operator-(const modint &a){return modint(M - a.val);}friend constexpr auto operator+(int64_t a, const modint &b){return modint(a) + b;}friend constexpr auto operator-(int64_t a, const modint &b){return modint(a) - b;}friend constexpr auto operator*(int64_t a, const modint &b){return modint(a) * b;}friend constexpr auto operator/(int64_t a, const modint &b){return modint(a) / b;}friend std::istream& operator>>(std::istream &s, modint<M> &a){s >> a.val; return s;}friend std::ostream& operator<<(std::ostream &s, const modint<M> &a){s << a.val; return s;}template <int N>static auto div(){static auto value = inv(N);return value;}explicit operator int32_t() const noexcept {return val;}explicit operator int64_t() const noexcept {return val;}};}/*** @title Basic graph* @docs graph.md*/namespace haar_lib {template <typename T>struct edge {int from, to;T cost;int index = -1;edge(){}edge(int from, int to, T cost): from(from), to(to), cost(cost){}edge(int from, int to, T cost, int index): from(from), to(to), cost(cost), index(index){}};template <typename T>struct graph {using weight_type = T;using edge_type = edge<T>;std::vector<std::vector<edge<T>>> data;auto& operator[](size_t i){return data[i];}const auto& operator[](size_t i) const {return data[i];}auto begin() const {return data.begin();}auto end() const {return data.end();}graph(){}graph(int N): data(N){}bool empty() const {return data.empty();}int size() const {return data.size();}void add_edge(int i, int j, T w, int index = -1){data[i].emplace_back(i, j, w, index);}void add_undirected(int i, int j, T w, int index = -1){add_edge(i, j, w, index);add_edge(j, i, w, index);}template <size_t I, bool DIRECTED = true, bool WEIGHTED = true>void read(int M){for(int i = 0; i < M; ++i){int u, v; std::cin >> u >> v;u -= I;v -= I;T w = 1;if(WEIGHTED) std::cin >> w;if(DIRECTED) add_edge(u, v, w, i);else add_undirected(u, v, w, i);}}};template <typename T>using tree = graph<T>;}/*** @title Heavy-light decomposition* @docs heavy_light_decomposition.md*/namespace haar_lib {template <typename T>class hl_decomposition {int n;std::vector<int> sub, // subtree sizepar, // parent idhead, // chain head idid, // id[original id] = hld idrid, // rid[hld id] = original idnext, // next node in a chainend; //int dfs_sub(tree<T> &tr, int cur, int p){par[cur] = p;int t = 0;for(auto &e : tr[cur]){if(e.to == p) continue;sub[cur] += dfs_sub(tr, e.to, cur);if(sub[e.to] > t){t = sub[e.to];next[cur] = e.to;std::swap(e, tr[cur][0]);}}return sub[cur];}void dfs_build(const tree<T> &tr, int cur, int &i){id[cur] = i;rid[i] = cur;++i;for(auto &e : tr[cur]){if(e.to == par[cur]) continue;head[e.to] = (e.to == tr[cur][0].to ? head[cur] : e.to);dfs_build(tr, e.to, i);}end[cur] = i;}public:hl_decomposition(tree<T> tr, int root):n(tr.size()), sub(n, 1), par(n, -1), head(n), id(n), rid(n), next(n, -1), end(n, -1){dfs_sub(tr, root, -1);int i = 0;dfs_build(tr, root, i);}std::vector<std::tuple<int, int, bool>> path_query_vertex(int x, int y) const {std::vector<std::tuple<int, int, bool>> ret;const int w = lca(x, y);{int y = w;bool d = true;while(1){if(id[x] > id[y]) std::swap(x, y), d = not d;int l = std::max(id[head[y]], id[x]), r = id[y] + 1;if(l != r) ret.emplace_back(l, r, d);if(head[x] == head[y]) break;y = par[head[y]];}}x = y;y = w;{std::vector<std::tuple<int, int, bool>> temp;bool d = false;while(1){if(id[x] > id[y]) std::swap(x, y), d = not d;int l = std::max({id[head[y]], id[x], id[w] + 1}), r = id[y] + 1;if(l != r) temp.emplace_back(l, r, d);if(head[x] == head[y]) break;y = par[head[y]];}std::reverse(temp.begin(), temp.end());ret.insert(ret.end(), temp.begin(), temp.end());}return ret;}std::vector<std::pair<int, int>> path_query_edge(int x, int y) const {std::vector<std::pair<int, int>> ret;while(1){if(id[x] > id[y]) std::swap(x, y);if(head[x] == head[y]){if(x != y) ret.emplace_back(id[x] + 1, id[y] + 1);break;}ret.emplace_back(id[head[y]], id[y] + 1);y = par[head[y]];}return ret;}std::pair<int, int> subtree_query_edge(int x) const {return {id[x] + 1, end[x]};}std::pair<int, int> subtree_query_vertex(int x) const {return {id[x], end[x]};}int get_edge_id(int u, int v) const { // 辺に対応するidif(par[u] == v) return id[u];if(par[v] == u) return id[v];return -1;}int parent(int x) const {return par[x];};int lca(int u, int v) const {while(1){if(id[u] > id[v]) std::swap(u, v);if(head[u] == head[v]) return u;v = par[head[v]];}}int get_id(int x) const {return id[x];}};}namespace haar_lib {template <typename T>class lazy_segment_tree_with_coefficients {const int depth, size, hsize;std::vector<T> data, lazy, coeff;void propagate(int i){if(lazy[i] == 0) return;if(i < hsize){lazy[i << 1 | 0] = lazy[i] + lazy[i << 1 | 0];lazy[i << 1 | 1] = lazy[i] + lazy[i << 1 | 1];}data[i] = data[i] + lazy[i] * coeff[i];lazy[i] = 0;}T update(int i, int l, int r, int s, int t, const T &x){propagate(i);if(r <= s || t <= l) return data[i];if(s <= l && r <= t){lazy[i] += x;propagate(i);return data[i];}return data[i] =update(i << 1 | 0, l, (l + r) / 2, s, t, x) +update(i << 1 | 1, (l + r) / 2, r, s, t, x);}T get(int i, int l, int r, int x, int y){propagate(i);if(r <= x || y <= l) return 0;if(x <= l && r <= y) return data[i];return get(i << 1 | 0, l, (l + r) / 2, x, y) + get(i << 1 | 1, (l + r) / 2, r, x, y);}public:lazy_segment_tree_with_coefficients(){}lazy_segment_tree_with_coefficients(int n, std::vector<T> coeff_):depth(n > 1 ? 32 - __builtin_clz(n - 1) + 1 : 1),size(1 << depth),hsize(size / 2),data(size, 0),lazy(size, 0),coeff(size, 0){for(int i = hsize; i < size; ++i) coeff[i] = coeff_[i - hsize];for(int i = hsize; --i >= 1;) coeff[i] = coeff[i << 1 | 0] + coeff[i << 1 | 1];}void update(int l, int r, const T &x){update(1, 0, hsize, l, r, x);}void update_at(int i, const T &x){update(i, i + 1, x);}T get(int l, int r){return get(1, 0, hsize, l, r);}T operator[](int i){return get(i, i + 1);}void init(const T &val){init_with_vector(std::vector<T>(hsize, val));}void init_with_vector(const std::vector<T> &val){data.assign(size, 0);lazy.assign(size, 0);for(int i = 0; i < (int)val.size(); ++i) data[hsize + i] = val[i];for(int i = hsize; --i >= 0;) data[i] = data[i << 1 | 0] + data[i << 1 | 1];}};}/*** @docs sort_simultaneously.md*/namespace haar_lib {template <typename Compare, typename ... Args>void sort_simultaneously(const Compare &compare, std::vector<Args> &... args){const int N = std::max({args.size() ...});std::vector<int> ord(N);std::iota(ord.begin(), ord.end(), 0);std::sort(ord.begin(), ord.end(), compare);(void)std::initializer_list<int>{(void([&](){auto temp = args;for(int i = 0; i < N; ++i) temp[i] = args[ord[i]];std::swap(temp, args);}()), 0) ...};}}namespace haar_lib {}namespace solver {using namespace haar_lib;constexpr int m1000000007 = 1000000007;constexpr int m998244353 = 998244353;void init(){std::cin.tie(0);std::ios::sync_with_stdio(false);std::cout << std::fixed << std::setprecision(12);std::cerr << std::fixed << std::setprecision(12);std::cin.exceptions(std::ios_base::failbit);}using mint = modint<m1000000007>;void solve(){int N; std::cin >> N;std::vector<mint> S(N); std::cin >> S;std::vector<mint> C(N); std::cin >> C;tree<int> tr(N);tr.read<1, false, false>(N - 1);auto hld = hl_decomposition(tr, 0);sort_simultaneously([&](int i, int j){return hld.get_id(i) < hld.get_id(j);}, S, C);auto seg = lazy_segment_tree_with_coefficients<mint>(N, C);seg.init_with_vector(S);int Q; std::cin >> Q;while(Q--){int type; std::cin >> type;if(type == 0){int X, Y, Z; std::cin >> X >> Y >> Z;--X, --Y;for(auto [l, r, d] : hld.path_query_vertex(X, Y)){seg.update(l, r, Z);}}else{int X, Y; std::cin >> X >> Y;--X, --Y;mint ans = 0;for(auto [l, r, d] : hld.path_query_vertex(X, Y)){ans += seg.get(l, r);}std::cout << ans << "\n";}}}}int main(){solver::init();while(true){try{solver::solve();}catch(const std::istream::failure &e){break;}}return 0;}