c++ - 为什么 std::sort 构造对象?

标签 c++ sorting swap move-constructor move-assignment-operator

<分区>

我创建了以下类来理解std::sort 的行为:

class X {
public:
  X(int i) : i_(i) { }
  X(X&& rhs) noexcept : i_(std::move(rhs.i_)) { mc_++; }
  X& operator=(X&& rhs) noexcept {
    i_ = std::move(rhs.i_); ao_++; return *this;
  }
  void swap(X& rhs) noexcept { std::swap(i_, rhs.i_); sw_++; } 
  friend bool operator<(const X& lhs, const X& rhs) {
    return lhs.i_ < rhs.i_;
  }
  static void reset() { mc_ = ao_ = sw_ = 0; }
private:
  int i_;
  static size_t mc_, ao_, sw_; // function-call counters
};
// void swap(X& lhs, X& rhs) { lhs.swap(rhs); }

并运行以下基准代码:

int main() {
  std::vector<X> v;
  for (int i = 0; i < 1000000; i++) v.emplace_back(i);

  std::mt19937 g(0xa41bc9); // fixed seed to compare measurements
  std::shuffle(v.begin(), v.end(), g);

  X::reset();
  std::sort(std::begin(v), std::end(v));
}

在线IDE中的全部代码在这里:https://wandbox.org/permlink/nbwRKptakgCSHK4f .

测量到的特定函数的调用次数如下(均带有-O2/O2标志):

   function:   move ctor    operator=        swap
  GCC 7.1.0:   5,007,335   11,700,048           0
clang 4.0.0:   4,932,061    9,973,899           0
 MSVC 19.11:   8,580,356   21,521,211           0

如果我取消注释 swap 函数,情况会好转:

   function:   move ctor    operator=        swap
  GCC 7.1.0:     999,999    3,685,376   4,007,336
clang 4.0.0:      72,554      254,885   4,859,507
 MSVC 19.11:     906,593    6,173,685   7,673,763

但是,移动构造函数(加析构函数)和移动赋值运算符的调用仍然很多。困扰我的是效率。例如,swapoperator= 的调用可以被编译器内联,但我猜编译器可能不会“内联”(优化掉)构造/对象的破坏

为什么要在排序中使用对象的构造/赋值?就地排序(通常是 std::sort 做的)可以完全通过比较和交换操作来实现.

更新

我的假设是错误的。优化对象创建/销毁似乎是完全合法的,例如:

X temp = std::move(x1);  
x1 = std::move(x2);
x2 = std::move(temp);

因此,这样的代码可以像自定义 swap 一样高效。在线示例:https://godbolt.org/g/ud4u9U - 没有移动构造函数/赋值运算符的调用,尽管这些都不是微不足道的,并且它们的功能内联到 main 中。

最佳答案

std::sort() 是一种混合算法。虽然普通的快速排序可以仅使用 swap() 操作(来自 std::partition()),但真正的排序方法很可能使用插入、堆和/或合并排序也是。对于这些排序算法,通常更有效的是将对象移开(通过移动构造),将对象移入当前“洞”,最后将不合适的对象移开。只保留一个临时对象可能是合理的,但很可能该算法使用了一些函数并且保留临时对象有点不切实际(尽管我之前没有考虑过并且可能值得尝试)。

今年早些时候我的 Quicker Sorting谈话记录在 Italian C++ Conference : 它详细介绍了快速排序的细节。

结果是:如果你想对对象进行排序,你最好确保复制/移动构造、复制/移动赋值、析构函数和 swap() 是快速的。我可以想象保留一个临时对象可以减少构造和销毁的需要,但任务将保留。专门的破坏性移动也可能会提高性能,但我还没有试验过。

关于c++ - 为什么 std::sort 构造对象?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45623856/

相关文章:

c++ - 移动指定索引处的数组元素

c++ - 我如何强类型定义非原始类型?

C++ "Bus error: 10"和使用指针

c++ - 获取有关函数地址的信息

arrays - 如何根据键的值对 TCL 数组进行排序?

java - 具有指定 z 索引值的最佳渲染绘制顺序函数

c++ - Eclipse:GNU 工具链 - 使用 g++ 编译的 C 文件,重复 GNU C

javascript - jQuery : swap two value and change style

linux - Perl 脚本异常行为/回收内存

c++ - 当使用自定义比较器交换相同类型的标准库容器时,为什么会发生此错误?