我将使用数组实现数组,并将图形实现为具有结构共享的单链表。也就是说,在通用 C++ 中,
#include <algorithm>
#include <iostream>
#include <memory>
#include <utility>
#include <vector>
template <typename T>
struct Node {
explicit Node(const T& e) : element(e) {}
T element;
std::size_t index = 0;
Node* previous = nullptr;
};
template <typename T, typename Compare>
std::vector<T> LongestIncreasingSubsequence(const std::vector<T>& elements,
Compare compare) {
if (elements.empty()) {
return {};
}
std::vector<std::unique_ptr<Node<T>>> node_ownership;
node_ownership.reserve(elements.size());
std::vector<Node<T>*> tableau;
for (const T& element : elements) {
auto node = std::make_unique<Node<T>>(element);
auto it = std::lower_bound(tableau.begin(), tableau.end(), node.get(),
[&](const Node<T>* a, const Node<T>* b) {
return compare(a->element, b->element);
});
if (it != tableau.begin()) {
auto previous = it[-1];
node->index = previous->index + 1;
node->previous = previous;
}
if (it != tableau.end()) {
*it = node.get();
} else {
tableau.push_back(node.get());
}
node_ownership.push_back(std::move(node));
}
Node<T>* longest = *std::max_element(
tableau.begin(), tableau.end(),
[](Node<T>* a, Node<T>* b) { return a->index < b->index; });
std::vector<T> result(longest->index + 1);
for (; longest != nullptr; longest = longest->previous) {
result.at(longest->index) = longest->element;
}
return result;
}
int main() {
for (int x : LongestIncreasingSubsequence(
std::vector<int>{3, 1, 4, 1, 5, 9, 2, 6, 5, 3, 5, 8, 9, 7, 9, 3},
std::less<int>())) {
std::cout << x << '\n';
}
}
如果你有幸使用垃圾收集的语言工作,你可以忽略和 的node_ownership
业务std::move
。
这是一个 Python 版本。
import bisect
def longest_increasing_subsequence(elements):
elements = list(elements)
if not elements:
return []
# Build the tableau
tableau_elements = []
tableau_indexes = []
predecessors = []
for i, element in enumerate(elements):
j = bisect.bisect_left(tableau_elements, element)
predecessors.append(tableau_indexes[j - 1] if j > 0 else None)
if j < len(tableau_elements):
tableau_elements[j] = element
tableau_indexes[j] = i
else:
tableau_elements.append(element)
tableau_indexes.append(i)
# Find the subsequence lengths
lengths = []
for i, predecessor in enumerate(predecessors):
lengths.append(1 if predecessor is None else lengths[predecessor] + 1)
# Extract the best subsequence
best = max(range(len(lengths)), key=lambda i: lengths[i])
subsequence = []
while best is not None:
subsequence.append(elements[best])
best = predecessors[best]
subsequence.reverse()
return subsequence
print(longest_increasing_subsequence([3, 1, 4, 1, 5, 9, 2, 6, 5, 3, 5, 8, 9, 7, 9, 3]))