| #include <bits/stdc++.h> |
| using namespace std; |
|
|
| int main() { |
| ios::sync_with_stdio(false); |
| cin.tie(nullptr); |
|
|
| int L, R; |
| cin >> L >> R; |
|
|
| |
| |
|
|
| struct Edge { int to, w; }; |
|
|
| |
| vector<vector<Edge>> adj; |
| int n = 0; |
| auto nn = [&]() -> int { adj.emplace_back(); return n++; }; |
|
|
| int END = nn(); |
| int START = nn(); |
|
|
| |
| map<int,int> fc; |
| fc[0] = END; |
| function<int(int)> getF = [&](int k) -> int { |
| if (fc.count(k)) return fc[k]; |
| int ch = getF(k-1), u = nn(); |
| adj[u].push_back({ch, 0}); |
| adj[u].push_back({ch, 1}); |
| return fc[k] = u; |
| }; |
|
|
| |
| map<tuple<int,int,int>,int> rc; |
| function<int(int,int,int)> gr = [&](int k, int lo, int hi) -> int { |
| if (lo > hi) return -1; |
| if (k == 0) return (lo == 0 && hi == 0) ? END : -1; |
| if (lo == 0 && hi == (1<<k)-1) return getF(k); |
| auto key = make_tuple(k, lo, hi); |
| auto it = rc.find(key); |
| if (it != rc.end()) return it->second; |
| int mid = 1 << (k-1); |
| int left = -1, right = -1; |
| if (lo <= min(hi, mid-1)) |
| left = gr(k-1, lo, min(hi, mid-1)); |
| if (max(lo, mid) <= hi) |
| right = gr(k-1, max(lo, mid) - mid, hi - mid); |
| if (left == -1 && right == -1) return rc[key] = -1; |
| int u = nn(); |
| if (left != -1) adj[u].push_back({left, 0}); |
| if (right != -1) adj[u].push_back({right, 1}); |
| return rc[key] = u; |
| }; |
|
|
| int lenL = 32 - __builtin_clz(L), lenR = 32 - __builtin_clz(R); |
| for (int len = lenL; len <= lenR; len++) { |
| int rs = 1 << (len-1), re = (1<<len)-1; |
| int cL = max(L, rs), cR = min(R, re); |
| if (cL > cR) continue; |
| int target; |
| if (len == 1) target = END; |
| else target = gr(len-1, cL - rs, cR - rs); |
| if (target != -1) |
| adj[START].push_back({target, 1}); |
| } |
|
|
| |
| vector<bool> reach(n, false); |
| { |
| queue<int> q; |
| q.push(START); |
| reach[START] = true; |
| while (!q.empty()) { |
| int u = q.front(); q.pop(); |
| for (auto& e : adj[u]) { |
| if (!reach[e.to]) { |
| reach[e.to] = true; |
| q.push(e.to); |
| } |
| } |
| } |
| } |
|
|
| |
| |
| |
|
|
| |
| |
| |
|
|
| |
| vector<int> depth(n, -1); |
| depth[END] = 0; |
| |
| |
| vector<int> indeg(n, 0); |
| for (int u = 0; u < n; u++) { |
| if (!reach[u]) continue; |
| for (auto& e : adj[u]) { |
| indeg[e.to]++; |
| } |
| } |
| vector<int> topo; |
| queue<int> q; |
| for (int u = 0; u < n; u++) { |
| if (reach[u] && indeg[u] == 0) q.push(u); |
| } |
| while (!q.empty()) { |
| int u = q.front(); q.pop(); |
| topo.push_back(u); |
| for (auto& e : adj[u]) { |
| if (--indeg[e.to] == 0) q.push(e.to); |
| } |
| } |
|
|
| |
| reverse(topo.begin(), topo.end()); |
|
|
| |
| |
| map<vector<pair<int,int>>, int> sig_to_rep; |
| vector<int> node_rep(n, -1); |
|
|
| for (int u : topo) { |
| if (!reach[u]) continue; |
| |
| vector<pair<int,int>> sig; |
| for (auto& e : adj[u]) { |
| sig.push_back({node_rep[e.to], e.w}); |
| } |
| sort(sig.begin(), sig.end()); |
|
|
| auto it = sig_to_rep.find(sig); |
| if (it != sig_to_rep.end()) { |
| node_rep[u] = it->second; |
| } else { |
| node_rep[u] = u; |
| sig_to_rep[sig] = u; |
| } |
| } |
|
|
| |
| |
| |
|
|
| |
| |
| |
|
|
| |
| set<int> reps; |
| for (int u : topo) { |
| if (reach[u]) reps.insert(node_rep[u]); |
| } |
|
|
| |
| map<int, int> new_id; |
| int cnt = 0; |
| new_id[node_rep[START]] = ++cnt; |
| for (int r : reps) { |
| if (r != node_rep[START]) { |
| new_id[r] = ++cnt; |
| } |
| } |
|
|
| |
| vector<vector<pair<int,int>>> out_adj(cnt + 1); |
| for (int r : reps) { |
| int id = new_id[r]; |
| for (auto& e : adj[r]) { |
| int child_rep = node_rep[e.to]; |
| out_adj[id].push_back({new_id[child_rep], e.w}); |
| } |
| |
| sort(out_adj[id].begin(), out_adj[id].end()); |
| out_adj[id].erase(unique(out_adj[id].begin(), out_adj[id].end()), out_adj[id].end()); |
| } |
|
|
| cout << cnt << "\n"; |
| for (int i = 1; i <= cnt; i++) { |
| cout << out_adj[i].size(); |
| for (auto& [to, w] : out_adj[i]) { |
| cout << " " << to << " " << w; |
| } |
| cout << "\n"; |
| } |
|
|
| return 0; |
| } |
|
|