mirror of
https://git.sb/baoshuo/OI-codes.git
synced 2025-01-26 18:20:09 +00:00
219 lines
4.0 KiB
C++
219 lines
4.0 KiB
C++
#include <iostream>
|
|
#include <algorithm>
|
|
#include <iterator>
|
|
#include <map>
|
|
#include <stack>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
using std::cin;
|
|
using std::cout;
|
|
const char endl = '\n';
|
|
|
|
const int N = 2e5 + 5;
|
|
|
|
int n, m, q;
|
|
int cnt = 1, edge_cnt;
|
|
std::map<int, int> map[N];
|
|
std::vector<std::pair<int, int>> qs[N];
|
|
std::vector<int> ans;
|
|
|
|
struct edge {
|
|
int l, r, x, y, w;
|
|
|
|
edge(const int &_x = 0,
|
|
const int &_y = 0,
|
|
const int &_l = 0,
|
|
const int &_w = 0)
|
|
: l(_l), r(0), x(_x), y(_y), w(_w) {}
|
|
} edges[N << 1];
|
|
|
|
struct dsu_item {
|
|
int f, w, s;
|
|
|
|
dsu_item(const int &_f = 0, const int &_w = 0)
|
|
: f(_f), w(_w), s(1) {}
|
|
} dsu[N];
|
|
|
|
struct seg_node : std::vector<edge> {
|
|
int l, r;
|
|
|
|
seg_node(const int &_l = 0,
|
|
const int &_r = 0,
|
|
const std::vector<edge> &_v = std::vector<edge>())
|
|
: std::vector<edge>(_v), l(_l), r(_r) {}
|
|
} tr[N << 2];
|
|
|
|
class liner_base {
|
|
private:
|
|
int b[31];
|
|
|
|
public:
|
|
void insert(int w) {
|
|
for (int i = 29; i >= 0; --i) {
|
|
if (!(w >> i)) continue;
|
|
|
|
if (!b[i]) {
|
|
b[i] = w;
|
|
|
|
break;
|
|
}
|
|
|
|
w ^= b[i];
|
|
}
|
|
}
|
|
|
|
int query(int w) {
|
|
for (int i = 29; i >= 0; --i) {
|
|
if (!((w >> i) & 1)) continue;
|
|
|
|
w ^= b[i];
|
|
}
|
|
|
|
return w;
|
|
}
|
|
} lb[std::__lg(N << 2) + 1];
|
|
|
|
int find(int x, int &w) {
|
|
while (x != dsu[x].f) {
|
|
w ^= dsu[x].w;
|
|
x = dsu[x].f;
|
|
}
|
|
|
|
return x;
|
|
}
|
|
|
|
void merge(int x, int y, int w, int d) {
|
|
if (x == y) {
|
|
lb[d].insert(w);
|
|
|
|
return;
|
|
}
|
|
|
|
if (dsu[x].s < dsu[y].s) std::swap(x, y);
|
|
|
|
dsu[y].f = x;
|
|
dsu[y].w = w;
|
|
dsu[x].s += dsu[y].s;
|
|
}
|
|
|
|
int query(int x, int y, int d) {
|
|
int w = 0;
|
|
|
|
x = find(x, w);
|
|
y = find(y, w);
|
|
|
|
if (x != y) return -1;
|
|
|
|
return lb[d].query(w);
|
|
}
|
|
|
|
void build(int u, int l, int r) {
|
|
tr[u] = seg_node(l, r);
|
|
|
|
if (l == r) return;
|
|
|
|
int mid = (l + r) >> 1;
|
|
|
|
build(u << 1, l, mid);
|
|
build(u << 1 | 1, mid + 1, r);
|
|
}
|
|
|
|
void insert(int u, const edge &e) {
|
|
if (e.l <= tr[u].l && tr[u].r <= e.r) {
|
|
tr[u].emplace_back(e);
|
|
|
|
return;
|
|
}
|
|
|
|
int mid = (tr[u].l + tr[u].r) >> 1;
|
|
|
|
if (e.l <= mid) insert(u << 1, e);
|
|
if (e.r > mid) insert(u << 1 | 1, e);
|
|
}
|
|
|
|
void solve(int u, int l, int r, int d) {
|
|
std::stack<std::pair<int, dsu_item>> st;
|
|
|
|
lb[d] = lb[d - 1];
|
|
|
|
for (auto e : tr[u]) {
|
|
int wx = 0, wy = 0;
|
|
int x = find(e.x, wx);
|
|
int y = find(e.y, wy);
|
|
|
|
st.emplace(x, dsu[x]);
|
|
st.emplace(y, dsu[y]);
|
|
|
|
merge(x, y, wx ^ wy ^ e.w, d);
|
|
}
|
|
|
|
if (l == r) {
|
|
for (auto o : qs[l]) {
|
|
ans.emplace_back(query(o.first, o.second, d));
|
|
}
|
|
} else {
|
|
int mid = (l + r) >> 1;
|
|
|
|
solve(u << 1, l, mid, d + 1);
|
|
solve(u << 1 | 1, mid + 1, r, d + 1);
|
|
}
|
|
|
|
while (!st.empty()) {
|
|
dsu[st.top().first] = st.top().second;
|
|
st.pop();
|
|
}
|
|
}
|
|
|
|
int main() {
|
|
std::ios::sync_with_stdio(false);
|
|
cin.tie(nullptr);
|
|
|
|
cin >> n >> m;
|
|
|
|
for (int i = 1; i <= n; i++) {
|
|
dsu[i] = dsu_item(i, 0);
|
|
}
|
|
|
|
for (int i = 1, x, y, d; i <= m; i++) {
|
|
cin >> x >> y >> d;
|
|
|
|
map[x][y] = map[y][x] = ++edge_cnt;
|
|
|
|
edges[edge_cnt] = edge(x, y, 1, d);
|
|
}
|
|
|
|
cin >> q;
|
|
|
|
for (int i = 1, op, x, y; i <= q; i++) {
|
|
cin >> op >> x >> y;
|
|
|
|
if (op == 1) {
|
|
int d;
|
|
|
|
cin >> d;
|
|
|
|
map[x][y] = map[y][x] = ++edge_cnt;
|
|
edges[edge_cnt] = edge(x, y, ++cnt, d);
|
|
} else if (op == 2) {
|
|
edges[map[x][y]].r = cnt++;
|
|
} else { // op == 3
|
|
qs[cnt].emplace_back(x, y);
|
|
}
|
|
}
|
|
|
|
build(1, 1, cnt);
|
|
|
|
for (int i = 1; i <= edge_cnt; i++) {
|
|
if (!edges[i].r) edges[i].r = cnt;
|
|
|
|
insert(1, edges[i]);
|
|
}
|
|
|
|
solve(1, 1, cnt, 1);
|
|
|
|
std::copy(ans.begin(), ans.end(), std::ostream_iterator<int>(cout, "\n"));
|
|
|
|
return 0;
|
|
}
|