結果

問題 No.235 めぐるはめぐる (5)
ユーザー pekempeypekempey
提出日時 2016-10-05 07:09:36
言語 C++11
(gcc 13.3.0)
結果
AC  
実行時間 533 ms / 10,000 ms
コード長 4,218 bytes
コンパイル時間 2,147 ms
コンパイル使用メモリ 188,780 KB
実行使用メモリ 26,688 KB
最終ジャッジ日時 2024-11-21 17:11:49
合計ジャッジ時間 4,796 ms
ジャッジサーバーID
(参考情報)
judge5 / judge2
このコードへのチャレンジ
(要ログイン)

テストケース

テストケース表示
入力 結果 実行時間
実行使用メモリ
testcase_00 AC 533 ms
25,516 KB
testcase_01 AC 320 ms
26,688 KB
testcase_02 AC 471 ms
25,844 KB
権限があれば一括ダウンロードができます
コンパイルメッセージ
main.cpp: In function ‘int in()’:
main.cpp:144:14: warning: ignoring return value of ‘int scanf(const char*, ...)’ declared with attribute ‘warn_unused_result’ [-Wunused-result]
  144 |         scanf("%d", &t);
      |         ~~~~~^~~~~~~~~~

ソースコード

diff #

#pragma GCC optimize ("O3")
#pragma GCC target ("avx")

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

const int mod = 1e9 + 7;
uint32_t inv;
int r2;

int reduce(uint64_t x) {
	uint64_t y = uint64_t(uint32_t(x) * inv) * mod;
	return int(x >> 32) + mod - int(y >> 32);
}

int transform(int n) {
	return reduce(int64_t(n) * r2);
}

int normalize(int n) {
	return n >= mod ? n - mod : n;
}

void init_montgomery_reduction() {
	inv = 1;
	for (int i = 0; i < 5; ++i) inv *= 2 - inv * uint32_t(mod);
	r2 = -uint64_t(mod) % mod;
}

void add(int &a, int b) {
	(a += b - mod) < 0 ? a += mod : a;
}

int modadd(int a, int b) {
	return (a += b - mod) < 0 ? a + mod : a;
}

int modmul(int a, int b) {
	return reduce(int64_t(a) * b);
}

struct HLDecomposition {
	vector<vector<int>> g;
	vector<int> vid, head, heavy, parent, len_path;

	HLDecomposition(int n) : g(n), vid(n, -1), head(n), heavy(n, -1), parent(n), len_path(n) {}

	void add(int u, int v) {
		g[u].push_back(v);
		g[v].push_back(u);
	}

	void build() {
		dfs(0, -1);
		bfs();
	}

	int dfs(int curr, int prev) {
		parent[curr] = prev;
		int sub = 1, max_sub = 0;
		for (int next : g[curr]) if (next != prev) {
			int sub_next = dfs(next, curr);
			sub += sub_next;
			if (max_sub < sub_next) max_sub = sub_next, heavy[curr] = next;
		}
		return sub;
	}

	void bfs() {
		int k = 0;
		queue<int> q({ 0 });
		while (!q.empty()) {
			int h = q.front(); q.pop();
			for (int i = h; i != -1; i = heavy[i]) {
				len_path[h]++;
				vid[i] = k++;
				head[i] = h;
				for (int j : g[i]) if (j != parent[i] && j != heavy[i]) q.push(j);
			}
		}
	}

	void for_each(int u, int v, function<void(int, int, int)> f) {
		if (vid[u] > vid[v]) swap(u, v);
		f(max(vid[head[v]], vid[u]), vid[v], len_path[head[v]]);
		if (head[u] != head[v]) for_each(u, parent[head[v]], f);
	}
};

const int N = 1 << 18;
struct node {
	int w, sum, add;
};
node seg[N * 2];

void push(int k) {
	if (seg[k].add == 0 || seg[k].add == mod) return;
	add(seg[k].sum, modmul(seg[k].add, seg[k].w));
	add(seg[k * 2 + 0].add, seg[k].add);
	add(seg[k * 2 + 1].add, seg[k].add);
	seg[k].add = 0;
}

void push_sup(int k, int H) {
	for (int i = H; i >= 1; i--) push(k >> i);
}

int get_val(int k) {
	return modadd(seg[k].sum, modmul(seg[k].add, seg[k].w));
}

void fix_sup(int k, int H) {
	for (int i = 1; i <= H; i++) {
		seg[k >> i].sum = modadd(get_val(k >> (i - 1)), get_val(k >> (i - 1) ^ 1));
	}
}

void update(int l, int r, int v, int H) {
	int ll = l;
	int rr = r;
	bool L = false;
	bool R = false;
	for (l += N, r += N; l < r; l >>= 1, r >>= 1) {
		if (l & 1) add(seg[l++].add, v), L = true;
		if (r & 1) add(seg[--r].add, v), R = true;
	}
	if (L) fix_sup(ll + N, H);
	if (R) fix_sup(rr + N - 1, H);
}

int query(int l, int r, int H) {
	push_sup(l + N, H);
	push_sup(r + N - 1, H);
	int res = 0;
	for (l += N, r += N; l < r; l >>= 1, r >>= 1) {
		if (l & 1) add(res, get_val(l++));
		if (r & 1) add(res, get_val(--r));
	}
	return res;
}

int in() {
	int t;
	scanf("%d", &t);
	return t;
}

int in_t() {
	return transform(in());
}

double elapsed() {
	static clock_t curr;
	clock_t prev = curr;
	curr = clock();
	return (double)(curr - prev) / CLOCKS_PER_SEC;
}

int main() {
	cerr << elapsed() << endl;
	init_montgomery_reduction();

	int n = in();

	vector<int> S(n), C(n);
	for (int i = 0; i < n; i++) S[i] = in_t();
	for (int i = 0; i < n; i++) C[i] = in_t();

	HLDecomposition hld(n);
	for (int i = 1; i < n; i++) hld.add(in() - 1, in() - 1);
	hld.build();

	for (int i = 0; i < n; i++) {
		seg[hld.vid[i] + N].sum = S[i];
		seg[hld.vid[i] + N].w = C[i];
	}
	for (int i = N - 1; i >= 1; i--) {
		seg[i].sum = modadd(seg[i * 2 + 0].sum, seg[i * 2 + 1].sum);
		seg[i].w = modadd(seg[i * 2 + 0].w, seg[i * 2 + 1].w);
	}

	cerr << elapsed() << endl;

	int q = in();
	while (q--) {
		int t = in();

		if (t == 0) {
			int u = in() - 1, v = in() - 1, z = in_t();
			hld.for_each(u, v, [&](int l, int r, int s) {
				update(l, r + 1, z, __lg(s * 2  - 1));
			});
		} else {
			int u = in() - 1, v = in() - 1;
			int ans = 0;
			hld.for_each(u, v, [&](int l, int r, int s) {
				add(ans, query(l, r + 1, __lg(s * 2 - 1)));
			});
			printf("%d\n", normalize(reduce(ans)));
		}
	}
	cerr << elapsed() << endl;
}
0