std::sort 平均和最坏时间复杂度均为 O(N log N),因主流实现采用内省排序:小数组用插入排序,中等规模用优化快排,递归深度超 2×⌊log₂N⌋ 时切至堆排序。
std::sort 在 C++ 标准中只要求平均时间复杂度为 O(N log N),不保证最坏情况仍是 O(N log N);但所有主流实现(GCC libstdc++、Clang libc++、MSVC STL)都采用**内省排序(introsort)**,因此实际最坏时间复杂度也是 O(N log N)。
这和纯 std::qsort(C 风格)或手写快排不同——后者若 pivot 选得差,可能退化到 O(N²)。而 std::sort 不会,它会在递归过深时自动切换策略。
常见误解:以为 std::sort 就是“快排”,其实它是个混合策略:
std::insertion_sort)2 × floor(log₂ N) 时,切到堆排序(std::make_heap + std::sort_heap)快排的递归深度直接关联 pivot 质量。最坏情况下(如已排序数组 + 固定首元素作 pivot),每次只减少一个元素,递归深度达 N,栈空间爆炸且性能崩坏。
内省排序用 log N 级别的深度阈值作为“安全阀”:
2 * std::__lg(__last - __first)(即 2 × floor(log₂ len))O(N log N) 且原地、稳定栈空间(O(1) 辅助空间)这个切换不是在运行时“检测是否变慢”,而是**纯深度计数**——不看数据分布,只防最坏递归路径。
不。这是个高频混淆点:std::sort 是不稳定的(equal elements 可能换位),而 std::stable_sort 保证稳定性,但它**不一定是堆排序**。
实际实现策略更复杂:
std::merge),若内存足够则分配临时缓冲区,否则降级为原地归并(代价高)所以 std::stable_sort 时间复杂度通常是 O(N log N),但空间复杂度可能是 O(N)(取决于是否能分配缓冲区)。它和 std::sort 的内省机制完全无关。
不影响核心流程(递归深度监控、算法切换逻辑),但会影响:
std::sort 对比较函数的要求:必须是 **strict weak ordering**,否则行为未定义(UB)——可能崩溃、死循环、或看似排好实则错乱一个典型错误是写成:[a, b] { return a —— 这违反 strict weak ordering(相等时返回 true,导致排序器误判等价关系),结果不可靠。
内省排序的精妙之处不在“多聪明”,而在“多保守”:它不赌数据分布,而是用可证的深度上界 + 确定性兜底,把最坏情况关进笼子。真正容易被忽略的是——你无法通过重载 operato