Submission #533782


Source Code Expand

#include <string>
#include <vector>
#include <algorithm>
#include <numeric>
#include <set>
#include <map>
#include <queue>
#include <iostream>
#include <sstream>
#include <cstdio>
#include <cmath>
#include <ctime>
#include <cstring>
#include <cctype>
#include <cassert>
#include <limits>
#include <functional>
#define rep(i,n) for(int (i)=0;(i)<(int)(n);++(i))
#define rer(i,l,u) for(int (i)=(int)(l);(i)<=(int)(u);++(i))
#define reu(i,l,u) for(int (i)=(int)(l);(i)<(int)(u);++(i))
#if defined(_MSC_VER) || __cplusplus > 199711L
#define aut(r,v) auto r = (v)
#else
#define aut(r,v) __typeof(v) r = (v)
#endif
#define each(it,o) for(aut(it, (o).begin()); it != (o).end(); ++ it)
#define all(o) (o).begin(), (o).end()
#define pb(x) push_back(x)
#define mp(x,y) make_pair((x),(y))
#define mset(m,v) memset(m,v,sizeof(m))
#define INF 0x3f3f3f3f
#define INFL 0x3f3f3f3f3f3f3f3fLL
using namespace std;
typedef vector<int> vi; typedef pair<int,int> pii; typedef vector<pair<int,int> > vpii; typedef long long ll;
template<typename T, typename U> inline void amin(T &x, U y) { if(y < x) x = y; }
template<typename T, typename U> inline void amax(T &x, U y) { if(x < y) x = y; }

struct HeavyLightDecomposition {
	vector<int> colors, positions;	//Vertex -> Color, Vertex -> Offset
	vector<int> lengths, parents, branches;	//Color -> Int, Color -> Color, Color -> Offset
	vector<int> parentnodes, depths;	//Vertex -> Vertex, Vertex -> Int
	//vector<FenwickTree>とかを避けて1次元にしたい時に使う
	//sortednodesの[lefts[v], rights[v])はvのsubtreeとなっている
	vector<int> sortednodes, offsets;	//Index -> Vertex, Color -> Index
	vector<int> lefts, rights;	//Vertex -> Index

	struct BuildDFSState {
		int i, len, parent;
		BuildDFSState() { }
		BuildDFSState(int i_, int l, int p): i(i_), len(l), parent(p) { }
	};

	//両方の辺があってもいいし、親から子への辺だけでもよい
	void build(const vector<vi> &g, int root) {
		int n = g.size();

		colors.assign(n, -1); positions.assign(n, -1);
		lengths.clear(); parents.clear(); branches.clear();
		parentnodes.assign(n, -1); depths.assign(n, -1);

		sortednodes.clear(); offsets.clear();
		lefts.assign(n, -1); rights.assign(n, -1);

		vector<int> subtreesizes;
		measure(g, root, subtreesizes);

		typedef BuildDFSState State;
		depths[root] = 0;
		vector<State> s;
		s.push_back(State(root, 0, -1));
		while(!s.empty()) {
			State t = s.back(); s.pop_back();
			int i = t.i, len = t.len;
			int index = sortednodes.size();
			int color = lengths.size();

			if(t.parent == -3) {
				rights[i] = index;
				continue;
			}

			if(t.parent != -2) {
				assert(parents.size() == color);
				parents.push_back(t.parent);
				branches.push_back(len);
				offsets.push_back(index);
				len = 0;
			}
			colors[i] = color;
			positions[i] = len;

			lefts[i] = index;
			sortednodes.push_back(i);

			int maxsize = -1, maxj = -1;
			each(j, g[i]) if(colors[*j] == -1) {
				if(maxsize < subtreesizes[*j]) {
					maxsize = subtreesizes[*j];
					maxj = *j;
				}
				parentnodes[*j] = i;
				depths[*j] = depths[i] + 1;
			}
			s.push_back(State(i, -1, -3));
			if(maxj == -1) {
				lengths.push_back(len + 1);
			}else {
				each(j, g[i]) if(colors[*j] == -1 && *j != maxj)
					s.push_back(State(*j, len, color));
				s.push_back(State(maxj, len + 1, -2));
			}
		}
	}
	
	void get(int v, int &c, int &p) const {
		c = colors[v]; p = positions[v];
	}
	bool go_up(int &c, int &p) const {
		p = branches[c]; c = parents[c];
		return c != -1;
	}

	inline const int *nodesBegin(int c) const { return &sortednodes[0] + offsets[c]; }
	inline const int *nodesEnd(int c) const { return &sortednodes[0] + (c+1 == offsets.size() ? sortednodes.size() : offsets[c+1]); }

private:
	void measure(const vector<vi> &g, int root, vector<int> &out_subtreesizes) const {
		out_subtreesizes.assign(g.size(), -1);
		vector<int> s;
		s.push_back(root);
		while(!s.empty()) {
			int i = s.back(); s.pop_back();
			if(out_subtreesizes[i] == -2) {
				int s = 1;
				each(j, g[i]) if(out_subtreesizes[*j] != -2)
					s += out_subtreesizes[*j];
				out_subtreesizes[i] = s;
			}else {
				s.push_back(i);
				each(j, g[i]) if(out_subtreesizes[*j] == -1)
					s.push_back(*j);
				out_subtreesizes[i] = -2;
			}
		}
	}
};

int lowest_common_ancestor(const HeavyLightDecomposition &hld, int x, int y) {
	int cx, px, cy, py;
	hld.get(x, cx, px);
	hld.get(y, cy, py);
	while(cx != cy) {
		if(hld.depths[*hld.nodesBegin(cx)] < hld.depths[*hld.nodesBegin(cy)])
			hld.go_up(cy, py);
		else
			hld.go_up(cx, px);
	}
	return hld.nodesBegin(cx)[min(px, py)];
}

void get_route(const HeavyLightDecomposition &hld, int u, int v, vector<pair<int,pii> > &route) {
	route.clear();
	int w = lowest_common_ancestor(hld, u, v), wc, wp;
	hld.get(w, wc, wp);
	rep(uv, 2) {
		int c, p;
		hld.get(uv == 0 ? u : v, c, p);
		int sz = route.size();
		while(1) {
			int top = c == wc ? wp + uv : 0;
			pii q = uv == 0 ? mp(p + 1, top) : mp(top, p + 1);
			if(q.first != q.second) {
				if(uv == 1 && c == wc) {	//cをユニークにするために
					assert(route[sz-1].first == c);
					assert(route[sz-1].second.first - route[sz-1].second.second == 1);
					assert(route[sz-1].second.first == q.first);
					route[sz-1].second = mp(route[sz-1].second.second, q.second);
				}else {
					route.push_back(mp(c, q));
				}
			}
			if(c == wc) break;
			hld.go_up(c, p);
		}
		if(uv == 1)
			reverse(route.begin() + sz, route.end());
	}
}


struct Node {
	static vector<Node> buf;
	static size_t bufp;
	typedef const Node *Ref;
	Ref l, r;
	int minPos, maxPos;
	Node(): l(NULL), r(NULL), minPos(INF), maxPos(-1) { }
	static Ref newNode(Ref l, Ref r, int minPos, int maxPos) {
		if(bufp == buf.size()) {
			cerr << "Memory exhausted!" << endl;
			abort();
		}
		Node *p = new(&buf[bufp ++])Node;
		p->l = l, p->r = r;
		p->minPos = minPos;
		p->maxPos = maxPos;
		return p;
	}
};
vector<Node> Node::buf;
size_t Node::bufp = 0;
typedef Node::Ref Ref;

Ref init(int left, int right) {
	if(left + 1 == right) {
		return Node::newNode(NULL, NULL, INF, -1);
	} else {
		int mid = (left + right) / 2;
		Ref l = init(left, mid);
		Ref r = init(mid, right);
		return Node::newNode(l, r, min(l->minPos, r->minPos), max(l->maxPos, r->maxPos));
	}
}

Ref insert(Ref p, int left, int right, int qi) {
	if(left + 1 == right) {
		assert(left == qi);
		return Node::newNode(NULL, NULL, qi, qi);
	} else {
		int mid = (left + right) / 2;
		Ref l = qi  < mid ? insert(p->l, left, mid, qi) : p->l;
		Ref r = mid <= qi ? insert(p->r, mid, right, qi) : p->r;
		return Node::newNode(l, r, min(l->minPos, r->minPos), max(l->maxPos, r->maxPos));
	}
}

int succ(Ref p, int left, int right, int ql, int qr) {
	if(ql <= left && right <= qr) {
		return p->minPos;
	} else {
		int mid = (left + right) / 2;
		int a = ql < mid ? succ(p->l, left, mid, ql, qr) : INF;
		int b = mid < qr ? succ(p->r, mid, right, ql, qr) : INF;
		return min(a, b);
	}
}

int pred(Ref p, int left, int right, int ql, int qr) {
	if(ql <= left && right <= qr) {
		return p->maxPos;
	} else {
		int mid = (left + right) / 2;
		int a = ql < mid ? pred(p->l, left, mid, ql, qr) : -1;
		int b = mid < qr ? pred(p->r, mid, right, ql, qr) : -1;
		return max(a, b);
	}
}

int main() {
	int n; int q;
	while(~scanf("%d%d", &n, &q)) {
		vector<int> a(n);
		for(int i = 0; i < n; ++ i)
			scanf("%d", &a[i]);
		vector<vi> g(n);
		rep(i, n - 1) {
			int b, c;
			scanf("%d%d", &b, &c), -- b, -- c;
			g[b].push_back(c);
			g[c].push_back(b);
		}
		HeavyLightDecomposition hld;
		hld.build(g, 0);
		vpii orda(n);
		rep(i, n) orda[i] = mp(a[i], i);
		sort(all(orda));
		Node::buf.resize(n * 20);
		Node::bufp = 0;
		Ref t = init(0, n);
		vector<Ref> ts(n+1);
		ts[0] = t;
		rep(ix, n) {
			int u = orda[ix].second, c, p;
			hld.get(u, c, p);
			int pos = hld.offsets[c] + p;
			t = insert(t, 0, n, pos);
			ts[ix + 1] = t;
		}
		vector<pair<int, pii> > route;
		rep(ii, q) {
			int x, u, v;
			scanf("%d%d%d", &x, &u, &v), -- u, -- v;
			get_route(hld, u, v, route);
			int curx = x;
			each(i, route) {
				int c = i->first, l = i->second.first, r = i->second.second;
				int o = hld.offsets[c];
				l += o, r += o;
				bool dir = l > r;
				if(dir) swap(l, r);
				while(l < r) {
					Ref t = ts[lower_bound(all(orda), mp(curx, INF)) - orda.begin()];
					int nextpos;
					if(!dir)
						nextpos = succ(t, 0, n, l, r);
					else
						nextpos = pred(t, 0, n, l, r);
					if(nextpos == -1 || nextpos == INF)
						break;
					int val = a[hld.sortednodes[nextpos]];
					assert(curx >= val);
					curx %= val;
					if(!dir)
						l = nextpos + 1;
					else
						r = nextpos;
				}
			}
			printf("%d\n", curx);
		}
	}
	return 0;
}

Submission Info

Submission Time
Task J - MODクエリ
User anta
Language C++11 (GCC 4.9.2)
Score 400
Code Size 9017 Byte
Status AC
Exec Time 1138 ms
Memory 60592 KB

Compile Error

./Main.cpp: In function ‘int main()’:
./Main.cpp:264:22: warning: ignoring return value of ‘int scanf(const char*, ...)’, declared with attribute warn_unused_result [-Wunused-result]
    scanf("%d", &a[i]);
                      ^
./Main.cpp:268:37: warning: ignoring return value of ‘int scanf(const char*, ...)’, declared with attribute warn_unused_result [-Wunused-result]
    scanf("%d%d", &b, &c), -- b, -- c;
                                     ^
./Main.cpp:292:43: warning: ignoring return value of ‘int scanf(const char*, ...)’, declared with attribute warn_unused_result [-Wunused-result]
    scanf("%d%d%d", &x, &u, &v), -- u, -- v;
                                           ^

Judge Result

Set Name Small All
Score / Max Score 30 / 30 370 / 370
Status
AC × 16
AC × 38
Set Name Test Cases
Small 00_sample_path_01.txt, 15_path_00.txt, 15_path_01.txt, 15_path_02.txt, 15_path_03.txt, 15_path_04.txt, 16_path_05.txt, 16_path_06.txt, 16_path_07.txt, 16_path_08.txt, 16_path_09.txt, 17_path_10.txt, 17_path_11.txt, 17_path_12.txt, 17_path_13.txt, 17_path_14.txt
All 00_sample_path_01.txt, 15_path_00.txt, 15_path_01.txt, 15_path_02.txt, 15_path_03.txt, 15_path_04.txt, 16_path_05.txt, 16_path_06.txt, 16_path_07.txt, 16_path_08.txt, 16_path_09.txt, 17_path_10.txt, 17_path_11.txt, 17_path_12.txt, 17_path_13.txt, 17_path_14.txt, 20_random_15.txt, 20_random_16.txt, 20_random_17.txt, 21_random_18.txt, 21_random_19.txt, 21_random_20.txt, 30_random_21.txt, 30_random_22.txt, 30_random_23.txt, 40_random_24.txt, 40_random_25.txt, 40_random_26.txt, 49_sample_02.txt, 61_test_27.txt, 61_test_28.txt, 61_test_29.txt, 63_star_30.txt, 63_star_31.txt, 63_star_32.txt, 64_three_33.txt, 64_three_34.txt, 64_three_35.txt
Case Name Status Exec Time Memory
00_sample_path_01.txt AC 27 ms 924 KB
15_path_00.txt AC 27 ms 740 KB
15_path_01.txt AC 27 ms 800 KB
15_path_02.txt AC 26 ms 912 KB
15_path_03.txt AC 25 ms 932 KB
15_path_04.txt AC 26 ms 928 KB
16_path_05.txt AC 26 ms 804 KB
16_path_06.txt AC 24 ms 800 KB
16_path_07.txt AC 26 ms 808 KB
16_path_08.txt AC 25 ms 800 KB
16_path_09.txt AC 26 ms 924 KB
17_path_10.txt AC 1094 ms 58356 KB
17_path_11.txt AC 1080 ms 58344 KB
17_path_12.txt AC 1090 ms 58356 KB
17_path_13.txt AC 1072 ms 58356 KB
17_path_14.txt AC 1104 ms 58356 KB
20_random_15.txt AC 26 ms 924 KB
20_random_16.txt AC 24 ms 800 KB
20_random_17.txt AC 26 ms 924 KB
21_random_18.txt AC 25 ms 928 KB
21_random_19.txt AC 25 ms 924 KB
21_random_20.txt AC 26 ms 804 KB
30_random_21.txt AC 24 ms 796 KB
30_random_22.txt AC 26 ms 800 KB
30_random_23.txt AC 26 ms 828 KB
40_random_24.txt AC 1138 ms 59148 KB
40_random_25.txt AC 1134 ms 58876 KB
40_random_26.txt AC 1121 ms 59148 KB
49_sample_02.txt AC 24 ms 924 KB
61_test_27.txt AC 552 ms 58912 KB
61_test_28.txt AC 542 ms 58892 KB
61_test_29.txt AC 552 ms 59144 KB
63_star_30.txt AC 710 ms 60592 KB
63_star_31.txt AC 710 ms 60528 KB
63_star_32.txt AC 716 ms 60528 KB
64_three_33.txt AC 1060 ms 58356 KB
64_three_34.txt AC 1087 ms 58240 KB
64_three_35.txt AC 1094 ms 58368 KB