官术网_书友最值得收藏!

Pitfalls with non-noexcept move constructors

Recall the implementation of vector::resize() from section Resizing a std::vector. When the vector resizes, it reallocates its underlying array and moves its elements into the new array--unless the element type is not "nothrow move-constructible," in which case it copies its elements! What this means is that resizing a vector of your own class type will be unnecessarily "pessimized" unless you go out of your way to specify that your move constructor is noexcept.

Consider the following class definitions:

    struct Bad {
int x = 0;
Bad() = default;
Bad(const Bad&) { puts("copy Bad"); }
Bad(Bad&&) { puts("move Bad"); }
};

struct Good {
int x = 0;
Good() = default;
Good(const Good&) { puts("copy Good"); }
Good(Good&&) noexcept { puts("move Good"); }
};

class ImplicitlyGood {
std::string x;
Good y;
};

class ImplicitlyBad {
std::string x;
Bad y;
};

We can test the behavior of these classes in isolation using a test harness such as the following. Running test() will print "copy Bad--move Good--copy Bad--move Good." What an appropriate mantra!

    template<class T>
void test_resizing()
{
std::vector<T> vec(1);
// Force a reallocation on the vector.
vec.resize(vec.capacity() + 1);
}

void test()
{
test_resizing<Good>();
test_resizing<Bad>();
test_resizing<ImplicitlyGood>();
test_resizing<ImplicitlyBad>();
}

This is a subtle and arcane point, but it can have a major effect on the efficiency of your C++ code in practice. A good rule of thumb is: Whenever you declare your own move constructor or swap function, make sure you declare it noexcept.

主站蜘蛛池模板: 富阳市| 田林县| 利川市| 蒲江县| 孝感市| 杭锦后旗| 涟水县| 富阳市| 措美县| 乡城县| 晴隆县| 无极县| 蓬莱市| 分宜县| 绵竹市| 教育| 泸定县| 马山县| 林西县| 云龙县| 迁安市| 安庆市| 榆树市| 岳西县| 永定县| 周宁县| 宣化县| 保德县| 海宁市| 淳化县| 阜平县| 普陀区| 广昌县| 城口县| 新田县| 永仁县| 中超| 德州市| 巴楚县| 邵武市| 汉中市|