c++中引入了右值引用移动语义,可以避免无谓的复制,提高程序性能。有点难理解,于是花时间整理一下自己的理解。

左值、右值

C++中所有的值都必然属于左值、右值二者之一。左值是指表达式结束后依然存在的持久化对象,右值是指表达式结束时就不再存在的临时对象。所有的具名变量或者对象都是左值,而右值不具名。很难得到左值和右值的真正定义,但是有一个可以区分左值和右值的便捷方法:看能不能对表达式取地址,如果能,则为左值,否则为右值

看见书上又将右值分为将亡值和纯右值。纯右值就是c++98标准中右值的概念,如非引用返回的函数返回的临时变量值;一些运算表达式,如1+2产生的临时变量;不跟对象关联的字面量值,如2,'c',true,"hello";这些值都不能够被取地址。

而将亡值则是c++11新增的和右值引用相关的表达式,这样的表达式通常时将要移动的对象、T&&函数返回值、std::move()函数的返回值等,

不懂将亡值和纯右值的区别其实没关系,统一看作右值即可,不影响使用。

示例:

int i=0;// i是左值, 0是右值class A {public:int a;
};
A getTemp()
{return A();
}
A a = getTemp();   // a是左值  getTemp()的返回值是右值(临时变量)

左值引用、右值引用

c++98中的引用很常见了,就是给变量取了个别名,在c++11中,因为增加了右值引用(rvalue reference)的概念,所以c++98中的引用都称为了左值引用(lvalue reference)

int a = 10;
int& refA = a; // refA是a的别名, 修改refA就是修改a, a是左值,左移是左值引用int& b = 1; //编译错误! 1是右值,不能够使用左值引用

c++11中的右值引用使用的符号是&&,如

int&& a = 1; //实质上就是将不具名(匿名)变量取了个别名
int b = 1;
int && c = b; //编译错误! 不能将一个左值复制给一个右值引用
class A {public:int a;
};
A getTemp()
{return A();
}
A && a = getTemp();   //getTemp()的返回值是右值(临时变量)

getTemp()返回的右值本来在表达式语句结束后,其生命也就该终结了(因为是临时变量),而通过右值引用,该右值又重获新生,其生命期将与右值引用类型变量a的生命期一样,只要a还活着,该右值临时变量将会一直存活下去。实际上就是给那个临时变量取了个名字。

注意:这里a类型是右值引用类型(int &&),但是如果从左值和右值的角度区分它,它实际上是个左值。因为可以对它取地址,而且它还有名字,是一个已经命名的右值。

所以,左值引用只能绑定左值,右值引用只能绑定右值,如果绑定的不对,编译就会失败。但是,常量左值引用却是个奇葩,它可以算是一个“万能”的引用类型,它可以绑定非常量左值、常量左值、右值,而且在绑定右值的时候,常量左值引用还可以像右值引用一样将右值的生命期延长,缺点是,只能读不能改。

const int & a = 1; //常量左值引用绑定 右值, 不会报错class A {public:int a;
};
A getTemp()
{return A();
}
const A & a = getTemp();   //不会报错 而 A& a 会报错

事实上,很多情况下我们用来常量左值引用的这个功能却没有意识到,如下面的例子:

#include <iostream>
using namespace std;class Copyable {
public:Copyable(){}Copyable(const Copyable &o) {cout << "Copied" << endl;}
};
Copyable ReturnRvalue() {return Copyable(); //返回一个临时对象
}
void AcceptVal(Copyable a) {}
void AcceptRef(const Copyable& a) {}int main() {cout << "pass by value: " << endl;AcceptVal(ReturnRvalue()); // 应该调用两次拷贝构造函数cout << "pass by reference: " << endl;AcceptRef(ReturnRvalue()); //应该只调用一次拷贝构造函数
}

当我敲完上面的例子并运行后,发现结果和我想象的完全不一样!期望AcceptVal(ReturnRvalue())需要调用两次拷贝构造函数,一次在ReturnRvalue()函数中,构造好了Copyable对象,返回的时候会调用拷贝构造函数生成一个临时对象,在调用AcceptVal()时,又会将这个对象拷贝给函数的局部变量a,一共调用了两次拷贝构造函数。而AcceptRef()的不同在于形参是常量左值引用,它能够接收一个右值,而且不需要拷贝。

而实际的结果是,不管哪种方式,一次拷贝构造函数都没有调用!

这是由于编译器默认开启了返回值优化(RVO/NRVO, RVO, Return Value Optimization 返回值优化,或者NRVO, Named Return Value Optimization)。编译器很聪明,发现在ReturnRvalue内部生成了一个对象,返回之后还需要生成一个临时对象调用拷贝构造函数,很麻烦,所以直接优化成了1个对象对象,避免拷贝,而这个临时变量又被赋值给了函数的形参,还是没必要,所以最后这三个变量都用一个变量替代了,不需要调用拷贝构造函数。

虽然各大厂家的编译器都已经都有了这个优化,但是这并不是c++标准规定的,而且不是所有的返回值都能够被优化,而这篇文章的主要讲的右值引用移动语义可以解决编译器无法解决的问题。

为了更好的观察结果,可以在编译的时候加上-fno-elide-constructors选项(关闭返回值优化)。

// g++ test.cpp -o test -fno-elide-constructors
pass by value:
Copied
Copied //可以看到确实调用了两次拷贝构造函数
pass by reference:
Copied

上面这个例子本意是想说明常量左值引用能够绑定一个右值,可以减少一次拷贝(使用非常量的左值引用会编译失败),但是顺便讲到了编译器的返回值优化。。编译器还是干了很多事情的,很有用,但不能过于依赖,因为你也不确定它什么时候优化了什么时候没优化。

总结一下,其中T是一个具体类型:

  1. 左值引用, 使用 T&, 只能绑定左值
  2. 右值引用, 使用 T&&, 只能绑定右值
  3. 常量左值, 使用 const T&, 既可以绑定左值又可以绑定右值
  4. 已命名的右值引用,编译器会认为是个左值
  5. 编译器有返回值优化,但不要过于依赖

移动构造和移动赋值

回顾一下如何用c++实现一个字符串类MyStringMyString内部管理一个C语言的char *数组,这个时候一般都需要实现拷贝构造函数和拷贝赋值函数,因为默认的拷贝是浅拷贝,而指针这种资源不能共享,不然一个析构了,另一个也就完蛋了。

具体代码如下:

解释一下下面代码中函数参数 const char* cstr=0 的意思

cstr指向的值不能被修改,但指针的值可以修改,例如,
const char* str = 0;
*str = 1;//编译到此处时出错,因为str指向的值不能被修改
str = &a; //(此处假定a为普通char型变量)这时却不出错,说明指针的值是可以被改变的
如果想要指针本身的值也不被改变,那么可以这样定义
char* const str = 0;//此处表示str不可改变
const char* const str = 0;//此处表示str本身不可被改变,而且他指向的值也不可被改变

#include <iostream>
#include <cstring>
#include <vector>
using namespace std;class MyString
{
public:static size_t CCtor; //统计调用拷贝构造函数的次数
//    static size_t CCtor; //统计调用拷贝构造函数的次数
public:// 构造函数MyString(const char* cstr=0){if (cstr) {m_data = new char[strlen(cstr)+1];strcpy(m_data, cstr);}else {m_data = new char[1];*m_data = '\0';}}// 拷贝构造函数MyString(const MyString& str) {CCtor ++;m_data = new char[ strlen(str.m_data) + 1 ];strcpy(m_data, str.m_data);}// 拷贝赋值函数 =号重载MyString& operator=(const MyString& str){if (this == &str) // 避免自我赋值!!return *this;delete[] m_data;m_data = new char[ strlen(str.m_data) + 1 ];strcpy(m_data, str.m_data);return *this;}~MyString() {delete[] m_data;}char* get_c_str() const { return m_data; }
private:char* m_data;
};
size_t MyString::CCtor = 0;int main()
{vector<MyString> vecStr;vecStr.reserve(1000); //先分配好1000个空间,不这么做,调用的次数可能远大于1000for(int i=0;i<1000;i++){vecStr.push_back(MyString("hello"));}cout << MyString::CCtor << endl;
}

代码看起来挺不错,却发现执行了1000次拷贝构造函数,如果MyString("hello")构造出来的字符串本来就很长,构造一遍就很耗时了,最后却还要拷贝一遍,而MyString("hello")只是临时对象,拷贝完就没什么用了,这就造成了没有意义的资源申请和释放操作,如果能够直接使用临时对象已经申请的资源,既能节省资源,又能节省资源申请和释放的时间。而C++11新增加的移动语义就能够做到这一点。

要实现移动语义就必须增加两个函数:移动构造函数和移动赋值构造函数。

#include <iostream>
#include <cstring>
#include <vector>
using namespace std;class MyString
{
public:static size_t CCtor; //统计调用拷贝构造函数的次数static size_t MCtor; //统计调用移动构造函数的次数static size_t CAsgn; //统计调用拷贝赋值函数的次数static size_t MAsgn; //统计调用移动赋值函数的次数public:// 构造函数MyString(const char* cstr=0){if (cstr) {m_data = new char[strlen(cstr)+1];strcpy(m_data, cstr);}else {m_data = new char[1];*m_data = '\0';}}// 拷贝构造函数MyString(const MyString& str) {CCtor ++;m_data = new char[ strlen(str.m_data) + 1 ];strcpy(m_data, str.m_data);}// 移动构造函数MyString(MyString&& str) noexcept:m_data(str.m_data) {MCtor ++;str.m_data = nullptr; //不再指向之前的资源了}// 拷贝赋值函数 =号重载MyString& operator=(const MyString& str){CAsgn ++;if (this == &str) // 避免自我赋值!!return *this;delete[] m_data;m_data = new char[ strlen(str.m_data) + 1 ];strcpy(m_data, str.m_data);return *this;}// 移动赋值函数 =号重载MyString& operator=(MyString&& str) noexcept{MAsgn ++;if (this == &str) // 避免自我赋值!!return *this;delete[] m_data;m_data = str.m_data;str.m_data = nullptr; //不再指向之前的资源了return *this;}~MyString() {delete[] m_data;}char* get_c_str() const { return m_data; }
private:char* m_data;
};
size_t MyString::CCtor = 0;
size_t MyString::MCtor = 0;
size_t MyString::CAsgn = 0;
size_t MyString::MAsgn = 0;
int main()
{vector<MyString> vecStr;vecStr.reserve(1000); //先分配好1000个空间for(int i=0;i<1000;i++){vecStr.push_back(MyString("hello"));}cout << "CCtor = " << MyString::CCtor << endl;cout << "MCtor = " << MyString::MCtor << endl;cout << "CAsgn = " << MyString::CAsgn << endl;cout << "MAsgn = " << MyString::MAsgn << endl;
}/* 结果
CCtor = 0
MCtor = 1000
CAsgn = 0
MAsgn = 0
*/

可以看到,移动构造函数与拷贝构造函数的区别是,拷贝构造的参数是const MyString& str,是常量左值引用,而移动构造的参数是MyString&& str,是右值引用,而MyString("hello")是个临时对象,是个右值,优先进入移动构造函数而不是拷贝构造函数。而移动构造函数与拷贝构造不同,它并不是重新分配一块新的空间,将要拷贝的对象复制过来,而是"偷"了过来,将自己的指针指向别人的资源,然后将别人的指针修改为nullptr,这一步很重要,如果不将别人的指针修改为空,那么临时对象析构的时候就会释放掉这个资源,"偷"也白偷了。下面这张图可以解释copy和move的区别。

copy和move的区别.png

不用奇怪为什么可以抢别人的资源,临时对象的资源不好好利用也是浪费,因为生命周期本来就是很短,在你执行完这个表达式之后,它就毁灭了,充分利用资源,才能很高效。

对于一个左值,肯定是调用拷贝构造函数了,但是有些左值是局部变量,生命周期也很短,能不能也移动而不是拷贝呢?C++11为了解决这个问题,提供了std::move()方法来将左值转换为右值,从而方便应用移动语义。我觉得它其实就是告诉编译器,虽然我是一个左值,但是不要对我用拷贝构造函数,而是用移动构造函数吧。。。

int main()
{vector<MyString> vecStr;vecStr.reserve(1000); //先分配好1000个空间for(int i=0;i<1000;i++){MyString tmp("hello");vecStr.push_back(tmp); //调用的是拷贝构造函数}cout << "CCtor = " << MyString::CCtor << endl;cout << "MCtor = " << MyString::MCtor << endl;cout << "CAsgn = " << MyString::CAsgn << endl;cout << "MAsgn = " << MyString::MAsgn << endl;cout << endl;MyString::CCtor = 0;MyString::MCtor = 0;MyString::CAsgn = 0;MyString::MAsgn = 0;vector<MyString> vecStr2;vecStr2.reserve(1000); //先分配好1000个空间for(int i=0;i<1000;i++){MyString tmp("hello");vecStr2.push_back(std::move(tmp)); //调用的是移动构造函数}cout << "CCtor = " << MyString::CCtor << endl;cout << "MCtor = " << MyString::MCtor << endl;cout << "CAsgn = " << MyString::CAsgn << endl;cout << "MAsgn = " << MyString::MAsgn << endl;
}/* 运行结果
CCtor = 1000
MCtor = 0
CAsgn = 0
MAsgn = 0CCtor = 0
MCtor = 1000
CAsgn = 0
MAsgn = 0
*/

下面再举几个例子:

MyString str1("hello"); //调用构造函数
MyString str2("world"); //调用构造函数
MyString str3(str1); //调用拷贝构造函数
MyString str4(std::move(str1)); // 调用移动构造函数、
//    cout << str1.get_c_str() << endl; // 此时str1的内部指针已经失效了!不要使用
//注意:虽然str1中的m_dat已经称为了空,但是str1这个对象还活着,知道出了它的作用域才会析构!而不是move完了立刻析构
MyString str5;
str5 = str2; //调用拷贝赋值函数
MyString str6;
str6 = std::move(str2); // str2的内容也失效了,不要再使用

需要注意一下几点:

  1. str6 = std::move(str2),虽然将str2的资源给了str6,但是str2并没有立刻析构,只有在str2离开了自己的作用域的时候才会析构,所以,如果继续使用str2m_data变量,可能会发生意想不到的错误。
  2. 如果我们没有提供移动构造函数,只提供了拷贝构造函数,std::move()会失效但是不会发生错误,因为编译器找不到移动构造函数就去寻找拷贝构造函数,也这是拷贝构造函数的参数是const T&常量左值引用的原因!
  3. c++11中的所有容器都实现了move语义,move只是转移了资源的控制权,本质上是将左值强制转化为右值使用,以用于移动拷贝或赋值,避免对含有资源的对象发生无谓的拷贝。move对于拥有如内存、文件句柄等资源的成员的对象有效,如果是一些基本类型,如int和char[10]数组等,如果使用move,仍会发生拷贝(因为没有对应的移动构造函数),所以说move对含有资源的对象说更有意义。

universal references(通用引用)

当右值引用和模板结合的时候,就复杂了。T&&并不一定表示右值引用,它可能是个左值引用又可能是个右值引用。例如:

template<typename T>
void f( T&& param){}
f(10);  //10是右值
int x = 10; //
f(x); //x是左值

如果上面的函数模板表示的是右值引用的话,肯定是不能传递左值的,但是事实却是可以。这里的&&是一个未定义的引用类型,称为universal references,它必须被初始化,它是左值引用还是右值引用却决于它的初始化,如果它被一个左值初始化,它就是一个左值引用;如果被一个右值初始化,它就是一个右值引用。

注意:只有当发生自动类型推断时(如函数模板的类型自动推导,或auto关键字),&&才是一个universal references

例如:

template<typename T>
void f( T&& param); //这里T的类型需要推导,所以&&是一个 universal referencestemplate<typename T>
class Test {Test(Test&& rhs); //Test是一个特定的类型,不需要类型推导,所以&&表示右值引用
};void f(Test&& param); //右值引用//复杂一点
template<typename T>
void f(std::vector<T>&& param); //在调用这个函数之前,这个vector<T>中的推断类型
//已经确定了,所以调用f函数的时候没有类型推断了,所以是 右值引用template<typename T>
void f(const T&& param); //右值引用
// universal references仅仅发生在 T&& 下面,任何一点附加条件都会使之失效

所以最终还是要看T被推导成什么类型,如果T被推导成了string,那么T&&就是string&&,是个右值引用,如果T被推导为string&,就会发生类似string& &&的情况,对于这种情况,c++11增加了引用折叠的规则,总结如下:

  1. 所有的右值引用叠加到右值引用上仍然使一个右值引用。
  2. 所有的其他引用类型之间的叠加都将变成左值引用。

如上面的T& &&其实就被折叠成了个string &,是一个左值引用。

#include <iostream>
#include <type_traits>
#include <string>
using namespace std;template<typename T>
void f(T&& param){if (std::is_same<string, T>::value)std::cout << "string" << std::endl;else if (std::is_same<string&, T>::value)std::cout << "string&" << std::endl;else if (std::is_same<string&&, T>::value)std::cout << "string&&" << std::endl;else if (std::is_same<int, T>::value)std::cout << "int" << std::endl;else if (std::is_same<int&, T>::value)std::cout << "int&" << std::endl;else if (std::is_same<int&&, T>::value)std::cout << "int&&" << std::endl;elsestd::cout << "unkown" << std::endl;
}int main()
{int x = 1;f(1); // 参数是右值 T推导成了int, 所以是int&& param, 右值引用f(x); // 参数是左值 T推导成了int&, 所以是int&&& param, 折叠成 int&,左值引用int && a = 2;f(a); //虽然a是右值引用,但它还是一个左值, T推导成了int&string str = "hello";f(str); //参数是左值 T推导成了string&f(string("hello")); //参数是右值, T推导成了stringf(std::move(str));//参数是右值, T推导成了string
}

所以,归纳一下, 传递左值进去,就是左值引用,传递右值进去,就是右值引用。如它的名字,这种类型确实很"通用",下面要讲的完美转发,就利用了这个特性。

完美转发

所谓转发,就是通过一个函数将参数继续转交给另一个函数进行处理,原参数可能是右值,可能是左值,如果还能继续保持参数的原有特征,那么它就是完美的。

void process(int& i){cout << "process(int&):" << i << endl;
}
void process(int&& i){cout << "process(int&&):" << i << endl;
}void myforward(int&& i){cout << "myforward(int&&):" << i << endl;process(i);
}int main()
{int a = 0;process(a); //a被视为左值 process(int&):0process(1); //1被视为右值 process(int&&):1process(move(a)); //强制将a由左值改为右值 process(int&&):0myforward(2);  //右值经过forward函数转交给process函数,却称为了一个左值,//原因是该右值有了名字  所以是 process(int&):2myforward(move(a));  // 同上,在转发的时候右值变成了左值  process(int&):0// forward(a) // 错误用法,右值引用不接受左值
}

上面的例子就是不完美转发,而c++中提供了一个std::forward()模板函数解决这个问题。将上面的myforward()函数简单改写一下:

void myforward(int&& i){cout << "myforward(int&&):" << i << endl;process(std::forward<int>(i));
}myforward(2); // process(int&&):2

上面修改过后还是不完美转发,myforward()函数能够将右值转发过去,但是并不能够转发左值,解决办法就是借助universal references通用引用类型和std::forward()模板函数共同实现完美转发。例子如下:

#include <iostream>
#include <cstring>
#include <vector>
using namespace std;void RunCode(int &&m) {cout << "rvalue ref" << endl;
}
void RunCode(int &m) {cout << "lvalue ref" << endl;
}
void RunCode(const int &&m) {cout << "const rvalue ref" << endl;
}
void RunCode(const int &m) {cout << "const lvalue ref" << endl;
}// 这里利用了universal references,如果写T&,就不支持传入右值,而写T&&,既能支持左值,又能支持右值
template<typename T>
void perfectForward(T && t) {RunCode(forward<T> (t));
}template<typename T>
void notPerfectForward(T && t) {RunCode(t);
}int main()
{int a = 0;int b = 0;const int c = 0;const int d = 0;notPerfectForward(a); // lvalue refnotPerfectForward(move(b)); // lvalue refnotPerfectForward(c); // const lvalue refnotPerfectForward(move(d)); // const lvalue refcout << endl;perfectForward(a); // lvalue refperfectForward(move(b)); // rvalue refperfectForward(c); // const lvalue refperfectForward(move(d)); // const rvalue ref
}

上面的代码测试结果表明,在universal referencesstd::forward的合作下,能够完美的转发这4种类型。

emplace_back减少内存拷贝和移动

我们之前使用vector一般都喜欢用push_back(),由上文可知容易发生无谓的拷贝,解决办法是为自己的类增加移动拷贝和赋值函数,但其实还有更简单的办法!就是使用emplace_back()替换push_back(),如下面的例子:

#include <iostream>
#include <cstring>
#include <vector>
using namespace std;class A {
public:A(int i){
//        cout << "A()" << endl;str = to_string(i);}~A(){}A(const A& other): str(other.str){cout << "A&" << endl;}public:string str;
};int main()
{vector<A> vec;vec.reserve(10);for(int i=0;i<10;i++){vec.push_back(A(i)); //调用了10次拷贝构造函数
//        vec.emplace_back(i);  //一次拷贝构造函数都没有调用过}for(int i=0;i<10;i++)cout << vec[i].str << endl;
}

可以看到效果是明显的,虽然没有测试时间,但是确实可以减少拷贝。emplace_back()可以直接通过构造函数的参数构造对象,但前提是要有对应的构造函数

对于mapset,可以使用emplace()。基本上emplace_back()对应push_bakc(), emplce()对应insert()

移动语义对swap()函数的影响也很大,之前实现swap可能需要三次内存拷贝,而有了移动语义后,就可以实现高性能的交换函数了。

template <typename T>
void swap(T& a, T& b)
{T tmp(std::move(a));a = std::move(b);b = std::move(tmp);
}

如果T是可移动的,那么整个操作会很高效,如果不可移动,那么就和普通的交换函数是一样的,不会发生什么错误,很安全。

总结

  • 由两种值类型,左值和右值。
  • 有三种引用类型,左值引用、右值引用和通用引用。左值引用只能绑定左值,右值引用只能绑定右值,通用引用由初始化时绑定的值的类型确定。
  • 左值和右值是独立于他们的类型的,右值引用可能是左值可能是右值,如果这个右值引用已经被命名了,他就是左值。
  • 引用折叠规则:所有的右值引用叠加到右值引用上仍然是一个右值引用,其他引用折叠都为左值引用。当T&&为模板参数时,输入左值,它将变成左值引用,输入右值则变成具名的右值应用。
  • 移动语义可以减少无谓的内存拷贝,要想实现移动语义,需要实现移动构造函数和移动赋值函数。
  • std::move()将一个左值转换成一个右值,强制使用移动拷贝和赋值函数,这个函数本身并没有对这个左值什么特殊操作。
  • std::forward()universal references通用引用共同实现完美转发。
  • empalce_back()替换push_back()增加性能。

TODO

  • 对模板类型自动推导还不太熟悉,继续学习Effective Modern C++。
  • std::move()和std::forward()好像实现的并不复杂,有机会弄明白实现原理。

我的SegmentFault链接

参考

  • 深入理解C++11:C++11新特性解析与应用
  • 深入应用C++11:代码优化与工程级应用
  • Effective Modern C++

[c++11]我理解的右值引用、移动语义和完美转发相关推荐

  1. C++右值引用与转移和完美转发

    C++右值引用与转移和完美转发 1.右值引用 1.1右值 lvalue 是 loactor value 的缩写,rvalue 是 read value 的缩写 左值是指存储在内存中.有明确存储地址(可 ...

  2. C++11新特性之左值右值及移动语句与完美转发

    C左值右值 左值和右值的由来 什么是左值和右值 左值右值的本质 引用 左值引用 右值引用 移动语句与完美转发 移动语句 实现移动构造函数和转移赋值函数 stdmove 完美转发Perfect Forw ...

  3. C++11新特性之右值引用

    什么是对象? An object is "something in memory". 什么是左值,什么是右值? An lvalue expression identifies a ...

  4. C++11 标准新特性: 右值引用与转移语义(点评)

    <<C++11 标准新特性: 右值引用与转移语义>> 原文地址如下 http://www.ibm.com/developerworks/cn/aix/library/1307_ ...

  5. std::move C++11 标准新特性: 右值引用与转移语义

    新特性的目的 右值引用 (Rvalue Referene) 是 C++ 新标准 (C++11, 11 代表 2011 年 ) 中引入的新特性 , 它实现了转移语义 (Move Sementics) 和 ...

  6. C++11 标准新特性: 右值引用与转移语义

    原文地址 http://www.ibm.com/developerworks/cn/aix/library/1307_lisl_c11/ C++ 的新标准 C++11 已经发布一段时间了.本文介绍了新 ...

  7. 【转】C++11 标准新特性: 右值引用与转移语义

    VS2013出来了,对于C++来说,最大的改变莫过于对于C++11新特性的支持,在网上搜了一下C++11的介绍,发现这篇文章非常不错,分享给大家同时自己作为存档. 原文地址:http://www.ib ...

  8. 深入浅出C++左值引用,右值引用,移动语义。

    什么是左值 右值? 简单来说左值就是可以取地址,在=左边的,而右值就是不可以取地址,在=右边的. int t=10; t可以通过&取地址在=左边 所以t是左值 10不可以取地址 在=右边10是 ...

  9. 2023最新C++面经(一):vector内存预分配,左值引用和右值引用,move语义

    文章目录 零.前言 一.在C++中,往vector插入1000个数字,怎么做能保证性能最高 二.在vector中对10000个数字删除偶数位置的数,怎么做保证性能较高 三.malloc用delete会 ...

  10. [C++11] 右值引用和移动语义

    c++11引入了右值引用和移动语义,通过避免无谓的复制,以提高程序的执行效率. 1.左值与右值 c++中的数值必属于左值或右值之一,通常有以下方法进行区分: 左值:在赋值语句左侧,右值:在赋值语句右侧 ...

最新文章

  1. 解决 mybatis-generator-maven-plugin 中 overwrite 配置无效的问题
  2. 幅度响应怎么计算_广播百科 频率响应
  3. Android深度探索第四章
  4. 有序的两个数组在满足其中一个数组的所有数都小于另外一个数组的情况下的整体的中位数
  5. Mock和Java单元测试中的Mock框架Mockito介绍
  6. 安全狗深圳主题沙龙会:企业如何打造云安全堡垒
  7. 不要被泛型和向后兼容性所愚弄。 使用泛型类型
  8. MATLAB1阶零模型,MATLAB 空间计量模型的实现
  9. 推荐系统走向下一阶段最重要的三个问题
  10. 电池供电的电容麦_板儿砖变电池?!是的,科学家已成功实现这一功能
  11. [转载] Java基础知识面试题(2020最新版)
  12. 天首发展,音飞储存要起飞
  13. python离线语音转文字_利用Python和API轻松实现:语音转文字,文字转语音
  14. 世界很大,一定要出去看看
  15. html表格背景图片格式,CSS常见样式(二)——列表,背景,边框,行高,表格,vertical-align...
  16. Python计算某年某月某日天数
  17. linux cat 颜色,使用lolcat为您的Linux终端带来彩虹般美丽的色彩
  18. 使用docker搭建个人博客
  19. 八皇后问题的进化(4)-python写的八皇后
  20. 餐厅订座系统如何选?餐厅订座系统推荐

热门文章

  1. 联想预装office2019_【联想电脑专属福利】超详细图文教程,手把手教你免费激活正版Office...
  2. java递归用for实现_用java实现的经典递归算法
  3. public 函数_C++:11友元函数、友元类
  4. 多个项目共用同一个redis_浅谈Redis分布式锁(上)
  5. 检查mysql数据库脚本_MYSQL 数据库状态检查脚本(Python版)
  6. linux vi 查找哈希,利用 Hashtable 实现快速查找比较-了解
  7. 浙江理工考研c语言程序设计,浙江理工大学C程序设计期末试卷A卷
  8. python生成pdf报告模块_python生成pdf报告、python实现html转换为pdf报告
  9. Git(3):git clone远程GitHub仓库代码出现“Permission Denied (publickey)”问题
  10. 【2019】微软Power BI 每月功能更新系列——Power BI 8月版本功能完整解读