From f79e19cf882a6e3fbbce6070ad86b25a9a861775 Mon Sep 17 00:00:00 2001 From: hitonanode <32937551+hitonanode@users.noreply.github.com> Date: Mon, 25 Oct 2021 23:23:00 +0900 Subject: [PATCH 1/2] Add permutation tree --- other_algorithms/permutation_tree.hpp | 137 ++++++++++++++++++ other_algorithms/permutation_tree.md | 89 ++++++++++++ .../test/permutation_tree.yuki1720.test.cpp | 48 ++++++ 3 files changed, 274 insertions(+) create mode 100644 other_algorithms/permutation_tree.hpp create mode 100644 other_algorithms/permutation_tree.md create mode 100644 other_algorithms/test/permutation_tree.yuki1720.test.cpp diff --git a/other_algorithms/permutation_tree.hpp b/other_algorithms/permutation_tree.hpp new file mode 100644 index 00000000..c3ca345f --- /dev/null +++ b/other_algorithms/permutation_tree.hpp @@ -0,0 +1,137 @@ +#pragma once +#include "../segmenttree/range-add-range-min.hpp" +#include +#include +#include +#include +#include + +// Permutation tree +// Complexity: O(N log N) +// https://codeforces.com/blog/entry/78898 https://yukicoder.me/problems/no/1720 +struct permutation_tree { + enum NodeType { + JoinAsc, + JoinDesc, + Cut, + Leaf, + None, + }; + struct node { + NodeType tp; + int L, R; // i in [L, R) + int mini, maxi; // A[i] in [mini, maxi] + std::vector child; + int sz() const { return R - L; } + template friend OStream &operator<<(OStream &os, const node &n) { + os << "[[" << n.L << ',' << n.R << ")(ch:"; + for (auto i : n.child) os << i << ','; + return os << ")(tp=" << n.tp << ")]"; + } + }; + + int root; + std::vector A; + std::vector nodes; + + void _add_child(int parid, int chid) { + nodes[parid].child.push_back(chid); + nodes[parid].L = std::min(nodes[parid].L, nodes[chid].L); + nodes[parid].R = std::max(nodes[parid].R, nodes[chid].R); + nodes[parid].mini = std::min(nodes[parid].mini, nodes[chid].mini); + nodes[parid].maxi = std::max(nodes[parid].maxi, nodes[chid].maxi); + } + + permutation_tree() : root(-1) {} + permutation_tree(const std::vector &A_) : root(-1), A(A_) { // A: nonempty perm., 0-origin + assert(!A.empty()); + RangeAddRangeMin seg((std::vector(A.size()))); + + std::vector hi{-1}, lo{-1}; + std::vector st; + for (int i = 0; i < int(A.size()); ++i) { + while (hi.back() >= 0 and A[i] > A[hi.back()]) { + seg.add(hi[hi.size() - 2] + 1, hi.back() + 1, A[i] - A[hi.back()]); + hi.pop_back(); + } + hi.push_back(i); + while (lo.back() >= 0 and A[i] < A[lo.back()]) { + seg.add(lo[lo.size() - 2] + 1, lo.back() + 1, A[lo.back()] - A[i]); + lo.pop_back(); + } + lo.push_back(i); + + int h = nodes.size(); + nodes.push_back({NodeType::Leaf, i, i + 1, A[i], A[i], std::vector{}}); + + while (true) { + NodeType join_tp = NodeType::None; + if (!st.empty() and nodes[st.back()].maxi + 1 == nodes[h].mini) join_tp = JoinAsc; + if (!st.empty() and nodes[h].maxi + 1 == nodes[st.back()].mini) join_tp = JoinDesc; + + if (!st.empty() and join_tp != NodeType::None) { + const node &vtp = nodes[st.back()]; + // Insert v as the child of the top node in the stack + if (join_tp == vtp.tp) { + // Append child to existing Join node + _add_child(st.back(), h); + h = st.back(); + st.pop_back(); + } else { + // Make new join node (with exactly two children) + int j = st.back(); + nodes.push_back( + {join_tp, nodes[j].L, nodes[j].R, nodes[j].mini, nodes[j].maxi, {j}}); + st.pop_back(); + _add_child(nodes.size() - 1, h); + h = nodes.size() - 1; + } + } else if (seg.prod(0, i + 1 - nodes[h].sz()) == 0) { + // Make Cut node + int L = nodes[h].L, R = nodes[h].R, maxi = nodes[h].maxi, mini = nodes[h].mini; + nodes.push_back({NodeType::Cut, L, R, mini, maxi, {h}}); + h = nodes.size() - 1; + do { + _add_child(h, st.back()); + st.pop_back(); + } while (nodes[h].maxi - nodes[h].mini + 1 != nodes[h].sz()); + std::reverse(nodes[h].child.begin(), nodes[h].child.end()); + } else { + break; + } + } + st.push_back(h); + seg.add(0, i + 1, -1); + } + assert(st.size() == 1); + root = st[0]; + } + + void to_DOT(std::string filename = "") const { + if (filename.empty()) filename = "permutation_tree_v=" + std::to_string(A.size()) + ".DOT"; + + std::ofstream ss(filename); + ss << "digraph{\n"; + int nleaf = 0; + for (int i = 0; i < int(nodes.size()); i++) { + ss << i << "[\n"; + std::string lbl; + if (nodes[i].tp == NodeType::Leaf) { + lbl = "A[" + std::to_string(nleaf) + "] = " + std::to_string(A[nleaf]), nleaf++; + } else { + lbl += std::string(nodes[i].tp == NodeType::Cut ? "Cut" : "Join") + "\\n"; + lbl += "[" + std::to_string(nodes[i].L) + ", " + std::to_string(nodes[i].R) + ")"; + } + ss << "label = \"" << lbl << "\",\n"; + ss << "]\n"; + for (const auto &ch : nodes[i].child) ss << i << " -> " << ch << ";\n"; + } + ss << "{rank = same;"; + for (int i = 0; i < int(nodes.size()); i++) { + if (nodes[i].tp == NodeType::Leaf) ss << ' ' << i << ';'; + } + ss << "}\n"; + ss << "}\n"; + ss.close(); + } +}; diff --git a/other_algorithms/permutation_tree.md b/other_algorithms/permutation_tree.md new file mode 100644 index 00000000..3597408f --- /dev/null +++ b/other_algorithms/permutation_tree.md @@ -0,0 +1,89 @@ +--- +title: Permutation tree (順列木) +documentation_of: ./permutation_tree.hpp +--- + +与えられた $[0, \dots, N - 1]$ の置換 $\mathbf{A} = [A\_0, \dots, A\_{N - 1}]$ について,この連続部分列であってその長さがそれに含まれる要素の最大値と最小値の差に $1$ を加えた値と等しくなるようなものを全て列挙するのに役立つデータ構造. + +Permutation tree は区間のマージ過程を表す木として表現される.$N$ 個の葉は長さ $1$ の区間(単一の要素)に対応し,根は $\mathbf{A}$ 全体に対応する. + +葉以外の全ての頂点は `Join` と `Cut` いずれかの属性を持つ.`Join` 属性を持つ頂点は,その子を $c\_1, \dots, c\_k$ とおくと,任意の $1 \le i \le j \le k$ について頂点 $(c\_i, \dots, c\_j)$ が表す区間の和集合は上記の条件を満たす.また,全ての頂点について,その頂点が表す区間全体は上記の条件を満たす.そして,上記の条件を満たす区間はこれらに限られるというのが最も重要な性質である. + + +## 使用方法(例) + +木の各頂点の情報はメンバ変数 `std::vector nodes` に格納されている.特に根が格納されている位置を示す変数が `tree.root`. + +```cpp +enum NodeType { + JoinAsc, // Join,特に A[i] の値が増加していく + JoinDesc, // Join,特に A[i] の値が減少していく + Cut, // Cut + Leaf, // 葉である + None, +}; +struct node { + NodeType tp; + int L, R; // [L, R) : 頂点が表す区間 + int mini, maxi; // 区間に含まれる A[i] (L <= i < R) の最小・最大値 + std::vector child; // 子の頂点番号(昇順) +}; +``` + +また.`to_DOT(std::string filename)` によって DOT 言語でのグラフ出力が可能. + +## 問題例 + +### [ZeptoLab Code Rush 2015 F. Pudding Monsters - Codeforces](https://codeforces.com/contest/526/problem/F) + +上記の条件を満たす区間の個数だけを求めればよい問題. + +```cpp +permutation_tree tree(A); + +auto rec = [&](auto &&self, int now) -> long long { + long long ret = 1; + const auto &v = tree.nodes[now]; + if (v.tp == permutation_tree::JoinAsc or v.tp == permutation_tree::JoinDesc) { + ret = (long long)v.child.size() * (v.child.size() - 1) / 2; + } + for (auto c : v.child) ret += self(self, c); + return ret; +}; +cout << rec(rec, tree.root) << '\n'; +``` + +### [No.1720 Division Permutation - yukicoder](https://yukicoder.me/problems/no/1720) + +木上を DFS しながら DP の遷移をさせていけばよく,これは以下のように再帰的に書ける. + +```cpp +permutation_tree tree(P); +vector dp(K + 1, vector(N + 1)); +dp[0][0] = 1; + +auto rec = [&](auto &&self, int now) -> void { + auto &v = tree.nodes[now]; + if (v.tp == permutation_tree::Cut or v.tp == permutation_tree::Leaf) { + for (int k = 0; k < K; ++k) dp[k + 1][v.R] += dp[k][v.L]; + } + + vector sum(K); + for (auto ch : v.child) { + self(self, ch); + if (v.tp == permutation_tree::JoinAsc or v.tp == permutation_tree::JoinDesc) { + for (int k = 0; k < K; ++k) { + dp[k + 1][tree.nodes[ch].R] += sum[k]; + sum[k] += dp[k][tree.nodes[ch].L]; + } + } + } +}; +rec(rec, tree.root); + +for (int i = 1; i <= K; i++) cout << dp[i][N].val() << '\n'; +``` + +## リンク + +1. [Tutorial on Permutation Tree (析合树) - Codeforces](https://codeforces.com/blog/entry/78898) diff --git a/other_algorithms/test/permutation_tree.yuki1720.test.cpp b/other_algorithms/test/permutation_tree.yuki1720.test.cpp new file mode 100644 index 00000000..e931c265 --- /dev/null +++ b/other_algorithms/test/permutation_tree.yuki1720.test.cpp @@ -0,0 +1,48 @@ +#define PROBLEM "https://yukicoder.me/problems/no/1720" +#include "../permutation_tree.hpp" +#include "../../modint.hpp" +#include + +using mint = ModInt<998244353>; +using namespace std; + +int N, K; +permutation_tree tree; +vector> dp; + +void rec(int now) { + const auto &v = tree.nodes[now]; + if (v.tp == permutation_tree::Cut or v.tp == permutation_tree::Leaf) { + for (int k = 0; k < K; ++k) dp[k + 1][v.R] += dp[k][v.L]; + } + + vector sum(K); + for (auto ch : v.child) { + rec(ch); + if (v.tp == permutation_tree::JoinAsc or v.tp == permutation_tree::JoinDesc) { + for (int k = 0; k < K; ++k) { + dp[k + 1][tree.nodes[ch].R] += sum[k]; + sum[k] += dp[k][tree.nodes[ch].L]; + } + } + } +}; + +int main() { + cin.tie(nullptr), ios::sync_with_stdio(false); + + cin >> N >> K; + vector P(N); + + for (auto &x : P) cin >> x; + for (auto &x : P) x--; + + tree = permutation_tree(P); + + dp.assign(K + 1, vector(N + 1)); + dp[0][0] = 1; + + rec(tree.root); + + for (int i = 1; i <= K; i++) cout << dp[i][N] << '\n'; +} From 08bff19a4c4a60900a9c3509b4bd44cfeae428aa Mon Sep 17 00:00:00 2001 From: GitHub Date: Mon, 25 Oct 2021 14:24:42 +0000 Subject: [PATCH 2/2] [auto-verifier] verify commit f79e19cf882a6e3fbbce6070ad86b25a9a861775 --- .verify-helper/timestamps.remote.json | 1 + 1 file changed, 1 insertion(+) diff --git a/.verify-helper/timestamps.remote.json b/.verify-helper/timestamps.remote.json index 02ca3b16..b25044f0 100644 --- a/.verify-helper/timestamps.remote.json +++ b/.verify-helper/timestamps.remote.json @@ -129,6 +129,7 @@ "number/test/sieve.stress.test.cpp": "2021-08-01 19:15:08 +0900", "number/test/sqrt_modint_runtime.test.cpp": "2021-06-06 17:00:00 +0900", "other_algorithms/test/enumerate_triangles.test.cpp": "2021-06-13 19:44:51 +0900", +"other_algorithms/test/permutation_tree.yuki1720.test.cpp": "2021-10-25 23:23:00 +0900", "other_algorithms/test/slope_trick_stress.test.cpp": "2021-09-18 11:42:06 +0900", "segmenttree/test/acl_range-affine-range-sum.test.cpp": "2021-06-06 14:54:00 +0900", "segmenttree/test/acl_rmq.test.cpp": "2021-02-13 02:36:53 +0900",