尝试使用 C++ 移动构造函数...但失败了
Trying to use c++ move constructor...and fail
就在我认为我了解 std::move 和移动构造函数的作用时,我尝试编写一些单元测试,实际测试一些 class...
的移动构造函数
令我惊讶的是,我发现我想不出一种方法来构造实际调用移动构造函数的代码。更糟糕的是,我什至无法在移动构造函数的主体中设置断点(在 VS2013 社区版中,调试,64 位构建)。
想知道这是否是编译器的特性,我使用 clang (3.4.1) 在我的 freebsd 虚拟机上编写了一些小的测试代码。在那里,我也未能找到一种方法来调用该移动构造函数。
#include <iostream>
#include <stdint.h>
#include <string>
#include <algorithm>
#include <functional>
#include <ctype.h>
#include <locale>
void InPlaceToUpper( std::string& target )
{
std::transform(target.begin(), target.end(), target.begin(), ::toupper);
}
void InPlaceToLower( std::string& target )
{
std::transform(target.begin(), target.end(), target.begin(), ::tolower);
}
std::string ToUpper( const std::string& s )
{
std::string result;
result.resize(s.length());
std::transform(s.begin(), s.end(), result.begin(), ::toupper);
return result;
}
std::string ToLower( const std::string& s)
{
std::string result;
result.resize(s.length());
std::transform(s.begin(), s.end(), result.begin(), ::tolower);
return result;
}
class CFoo
{
std::string m_value;
public:
CFoo()
: m_value()
{
std::cout << "CFoo() called." << std::endl;
}
CFoo(const char *value)
: m_value(value)
{
std::cout << "CFoo(const char *) called." << std::endl;
}
CFoo(const std::string& value )
: m_value(value)
{
std::cout << "CFoo(const std::string&) called." << std::endl;
}
CFoo(const CFoo& other )
: m_value(other.m_value)
{
std::cout << "CFoo() copy constructor called." << std::endl;
}
CFoo(CFoo&& other )
: m_value(std::move(other.m_value))
{
std::cout << "CFoo() move constructor called." << std::endl;
std::cout << "other.m_value = " << other.m_value.c_str() << std::endl;
}
~CFoo()
{
std::cout << "~CFoo() called." << std::endl;
}
const CFoo& operator=( const CFoo& other )
{
std::cout << "CFoo copy assignment operator called." << std::endl;
if( &other != this )
{
m_value = other.m_value;
}
return *this;
}
const CFoo& operator=( CFoo&& other )
{
std::cout << "CFoo move assignment operator called." << std::endl;
if( &other != this )
{
m_value = std::move(other.m_value);
}
return *this;
}
CFoo ToUpper()
{
return CFoo(::ToUpper(m_value));
}
CFoo ToLower()
{
return CFoo(::ToLower(m_value));
}
const char * ToString() const
{
return m_value.c_str();
}
};
int main( int argc, const char *argv[] )
{
{
CFoo foo;
CFoo foo1("Hello World");
CFoo foo2 = CFoo("Hello again World!");
CFoo foo3(CFoo("Bye world"));
CFoo foo4 = CFoo("Bye again world");
CFoo foo5 = foo4.ToUpper();
CFoo foo6 = foo4.ToLower();
foo6 = foo4.ToUpper();
std::cout << "foo4: " << foo4.ToString() << std::endl;
foo6 = CFoo("Well well well");
}
return 0;
}
如果代码没有尽可能短,我深表歉意。但是只有几个地方要看,即我在 main() 中调用移动构造函数的努力以及 class Foo.
中各种构造函数的定义。
我知道允许关闭 RVO 和其他东西的编译器设置,但是为了在性能感知代码中使用特性 "move constructor",应该有一个 classic 示例何时它被调用。如果不是这样,我可能会决定根本不使用移动构造函数。
为了回答这个问题,你可以告诉我我可以在 main() 中写的一行,它调用了 CFoo 的移动构造函数。或者你可以告诉我我做错了什么。
std::string支持这样移动吗?也许这就是我的努力失败的原因?
提前致谢。
首先,std::string ToUpper(const std::string& s)
中有一个错误,即result
中没有space。 C++ 算法不会自动增长其目标容器。您必须自己分配 space,或者使用插入器适配器。
让 space 进出,例如这样做:
result.resize(s.length());
之后移动赋值运算符调用:
foo6 = foo4.ToUpper();
foo6 = CFoo("Well well well");
每当从 xvalue 相同类型的对象初始化时调用移动构造函数,其中包括:
- 初始化,
T a = std::move(b);
或 T a(std::move(b));
,其中 b
是类型 T
- 函数参数传递:
f(std::move(a));
,其中 a
是 T
类型,f
是 void f(T t)
- function return:
return a;
在函数内部,例如 T f()
,其中 a
是 T
类型,它有一个移动构造函数。
有关更多信息,请参阅 cppreference.com 上的 move constructors。
在您使用移动构造函数的所有尝试中,它都是 elided,例如CFoo foo = CFoo(blah);
等同于 CFoo foo(blah);
不需要使用移动构造函数。这是一件好事,因为编译器正在优化根本不需要进行任何复制或移动。
要查看正在使用的移动构造函数,请尝试:
CFoo f1;
CFoo f2 = std::move(f1);
这从右值构造f2
,不能省略任何内容,因此将使用移动构造函数。
就在我认为我了解 std::move 和移动构造函数的作用时,我尝试编写一些单元测试,实际测试一些 class...
的移动构造函数令我惊讶的是,我发现我想不出一种方法来构造实际调用移动构造函数的代码。更糟糕的是,我什至无法在移动构造函数的主体中设置断点(在 VS2013 社区版中,调试,64 位构建)。
想知道这是否是编译器的特性,我使用 clang (3.4.1) 在我的 freebsd 虚拟机上编写了一些小的测试代码。在那里,我也未能找到一种方法来调用该移动构造函数。
#include <iostream>
#include <stdint.h>
#include <string>
#include <algorithm>
#include <functional>
#include <ctype.h>
#include <locale>
void InPlaceToUpper( std::string& target )
{
std::transform(target.begin(), target.end(), target.begin(), ::toupper);
}
void InPlaceToLower( std::string& target )
{
std::transform(target.begin(), target.end(), target.begin(), ::tolower);
}
std::string ToUpper( const std::string& s )
{
std::string result;
result.resize(s.length());
std::transform(s.begin(), s.end(), result.begin(), ::toupper);
return result;
}
std::string ToLower( const std::string& s)
{
std::string result;
result.resize(s.length());
std::transform(s.begin(), s.end(), result.begin(), ::tolower);
return result;
}
class CFoo
{
std::string m_value;
public:
CFoo()
: m_value()
{
std::cout << "CFoo() called." << std::endl;
}
CFoo(const char *value)
: m_value(value)
{
std::cout << "CFoo(const char *) called." << std::endl;
}
CFoo(const std::string& value )
: m_value(value)
{
std::cout << "CFoo(const std::string&) called." << std::endl;
}
CFoo(const CFoo& other )
: m_value(other.m_value)
{
std::cout << "CFoo() copy constructor called." << std::endl;
}
CFoo(CFoo&& other )
: m_value(std::move(other.m_value))
{
std::cout << "CFoo() move constructor called." << std::endl;
std::cout << "other.m_value = " << other.m_value.c_str() << std::endl;
}
~CFoo()
{
std::cout << "~CFoo() called." << std::endl;
}
const CFoo& operator=( const CFoo& other )
{
std::cout << "CFoo copy assignment operator called." << std::endl;
if( &other != this )
{
m_value = other.m_value;
}
return *this;
}
const CFoo& operator=( CFoo&& other )
{
std::cout << "CFoo move assignment operator called." << std::endl;
if( &other != this )
{
m_value = std::move(other.m_value);
}
return *this;
}
CFoo ToUpper()
{
return CFoo(::ToUpper(m_value));
}
CFoo ToLower()
{
return CFoo(::ToLower(m_value));
}
const char * ToString() const
{
return m_value.c_str();
}
};
int main( int argc, const char *argv[] )
{
{
CFoo foo;
CFoo foo1("Hello World");
CFoo foo2 = CFoo("Hello again World!");
CFoo foo3(CFoo("Bye world"));
CFoo foo4 = CFoo("Bye again world");
CFoo foo5 = foo4.ToUpper();
CFoo foo6 = foo4.ToLower();
foo6 = foo4.ToUpper();
std::cout << "foo4: " << foo4.ToString() << std::endl;
foo6 = CFoo("Well well well");
}
return 0;
}
如果代码没有尽可能短,我深表歉意。但是只有几个地方要看,即我在 main() 中调用移动构造函数的努力以及 class Foo.
中各种构造函数的定义。我知道允许关闭 RVO 和其他东西的编译器设置,但是为了在性能感知代码中使用特性 "move constructor",应该有一个 classic 示例何时它被调用。如果不是这样,我可能会决定根本不使用移动构造函数。
为了回答这个问题,你可以告诉我我可以在 main() 中写的一行,它调用了 CFoo 的移动构造函数。或者你可以告诉我我做错了什么。
std::string支持这样移动吗?也许这就是我的努力失败的原因?
提前致谢。
首先,std::string ToUpper(const std::string& s)
中有一个错误,即result
中没有space。 C++ 算法不会自动增长其目标容器。您必须自己分配 space,或者使用插入器适配器。
让 space 进出,例如这样做:
result.resize(s.length());
之后移动赋值运算符调用:
foo6 = foo4.ToUpper();
foo6 = CFoo("Well well well");
每当从 xvalue 相同类型的对象初始化时调用移动构造函数,其中包括:
- 初始化,
T a = std::move(b);
或T a(std::move(b));
,其中b
是类型T
- 函数参数传递:
f(std::move(a));
,其中a
是T
类型,f
是void f(T t)
- function return:
return a;
在函数内部,例如T f()
,其中a
是T
类型,它有一个移动构造函数。
有关更多信息,请参阅 cppreference.com 上的 move constructors。
在您使用移动构造函数的所有尝试中,它都是 elided,例如CFoo foo = CFoo(blah);
等同于 CFoo foo(blah);
不需要使用移动构造函数。这是一件好事,因为编译器正在优化根本不需要进行任何复制或移动。
要查看正在使用的移动构造函数,请尝试:
CFoo f1;
CFoo f2 = std::move(f1);
这从右值构造f2
,不能省略任何内容,因此将使用移动构造函数。