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

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.

主站蜘蛛池模板: 钟山县| 平和县| 永清县| 武强县| 漯河市| 灵璧县| 和龙市| 高州市| 文成县| 罗平县| 佛学| 米易县| 克什克腾旗| 江门市| 阿拉善左旗| 揭东县| 饶阳县| 磐安县| 万州区| 海门市| 黑河市| 武邑县| 庄浪县| 堆龙德庆县| 堆龙德庆县| 伊通| 江陵县| 福州市| 正定县| 玉屏| 莫力| 贵港市| 阜平县| 石首市| 赤城县| 玉溪市| 新郑市| 西丰县| 襄汾县| 山东| 襄汾县|