网站怎么更改域名解析,wordpress页面链接404,昆明岭蓝科技,网站是不是要用代码做左值引用和右值引用
回顾引用
我们之前就了解到了左值引用#xff0c;首先我们要了解引用在编译器底层其实就是指针。具体来说#xff0c;当声明引用时#xff0c;编译器会在底层生成一个指针来表示引用#xff0c;但在代码编写和使用时#xff0c;我们可以像使用变量类…左值引用和右值引用
回顾引用
我们之前就了解到了左值引用首先我们要了解引用在编译器底层其实就是指针。具体来说当声明引用时编译器会在底层生成一个指针来表示引用但在代码编写和使用时我们可以像使用变量类似取别名的方式一样来操作引用而不需要显式地使用指针符号。这使得引用更为方便且看起来更直观同时也能保证所获得的引用总是有效的。那什么是左值什么又是右值呢 什么是左值与右值
左值是表示数据的表达式(如变量名或解引用的指针)也可能在赋值符号右边具有地址的、可寻址的表达式才是左值。typename
右值是一个表达式如字面常量、表达式返回值函数返回值等等且右值不能出现在赋值符号左边不能进行取地址才是右值。typename 右值、左值引用的使用
//左值引用右值
int main()
{double x 1.1, y 2.2;double r1 x;//double r2 x y;//此时的x1x2是右值必须加const修饰才能引用const double r2 x y;return 0;
}
1. 左值引用只能引用左值不能引用右值。 2. 但是const左值引用既可引用左值也可引用右值。 //右值引用左值
int main()
{double x 1.1, y 2.2;double rr1 xy;//double r2 x;//此时的x1是左值所以要将属性转换再引用double r2 move(x);return 0;
}
1. 右值引用只能右值不能引用左值。 2. 但是右值引用可以move以后的左值。此时改变的其实是返回值属性 移动构造和移动赋值 引用的好处就是不需要进行拷贝所以对于自定义类型而言引用的好处就显得更为重要。那么右值引用的好处也是同样......
下面就简单的以my_string的构造相关函数进行举例
class my_string
{
public://构造函数my_string(const char* tmp):_size(strlen(tmp)), _capacity(_size){_s new char[_capacity 1];//还有一个\0strcpy(_s, tmp);}// 拷贝构造my_string(const my_string s):_s(nullptr){cout my_string(const my_string s) -- 深拷贝 endl;}// 赋值重载my_string operator(const my_string s){cout my_string operator(my_string s) -- 深拷贝 endl;}private:char* _s;size_t _size;size_t _capacity;
};以to_string函数举例深拷贝的消耗
my_string to_string(int num)
{my_string ret;//.........//对ret操作return ret;//ret返回值会拷贝构造临时对象
}
int main()
{my_string s;s to_string(1234);//调用赋值重载}
其实经过分析我们知道to_string函数内部创建了一个对象所以在函数调用结束的时候就会释放该ret对象的空间所以当我们返回的时候实质上是将ret存进一个临时对象中作为返回值所以此时会发生第一次的深拷贝最后返回值再赋值给s对象所以又发生了一次深拷贝。此时就相当于一共复制拷贝了两份空间再算上原来的就有三份空间所以此时如果该空间不是存的to_string而是存的是其他的大量数据的话那么此时的消耗是极其大的。
此时其实我们要了解C11中将右值分为两类1.纯右值内置类型的右值2.将亡值自定义类型的右值
而此时我们的ret出了作用域就销毁而且还会创建临时对象所以我们的ret其实会被视作将亡值所以此时就可以看作是一个右值编译器进行的特殊处理所以我们就可以不用再调用拷贝构造来创建临时对象了呀反正该对象也要销毁何不直接将临时对象指向该将亡值呢。同理再进行赋值的时候同样也可以啊函数返回值就是典型的右值临时对象 所以就引出了移动构造和移动拷贝 void swap(my_string s){std::swap(_s, s._s);std::swap(_size, s._size);std::swap(_capacity, s._capacity);}// 移动构造my_string(my_string s)//此时编译器会选择更匹配的而不是const my_string:_s(nullptr), _size(0), _capacity(0){swap(s);}// 移动赋值my_string operator(my_string s){swap(s);return *this;}
但是我们要注意的点就是当我们实现了移动拷贝和移动赋值以后那么我们将对象move修饰之后进行拷贝和赋值以后会进行swap进行资源的转移所以会将该move对象的资源交换所以在进行移动拷贝之后该move对象的资源交换后就没了而调用移动赋值之后move对象的资源就是交换了。
所以最好不要随意地对一个左值进行move修饰可能会有意想不到的结果发生。 其实C11之后对于容器的push函数都增加了移动构造的形式这种尤其是对那些需要进行深拷贝的对象而言会方便很多。此时就不需要再拷贝一份直接将资源交换即可完成拷贝工作。 右值引用后的属性 右值被右值引用之后的属性是左值所以就可以被修改
int main()
{double x 1.1, y 2.2;int rr1 10;const double rr2 x y;rr1 20;//右值引用后可以修改rr2 5.5; // 报错return 0;
}
实例 就该移动构造而言s对象就是右值引用的过后的再调用swap函数此时在string s接受参数的时候其实s对象的属性就变成了左值即可被修改所以传给swap函数时用string s接受就没问题但是两个过程中自始至终依旧是一个对象没有创建额外的对象。 就拿这上述string的push_back函数来说我们知道C11之后容器的push_back函数都重载了一份右值插入。所以对于以上的函数来说就显得很麻烦了因为我们知道右值引用接收右值以后数据的属性就会变成左值因此对于这种函数嵌套调用的情况就会显得十分繁琐需要将每次参数都move一下才可以实现接下来的右值引用接收。但是我们move之后就相当于是写固定了所以相较于每一次的move对象我们用完美转发会更好。 完美转发 void Fun(int x){ cout 左值引用 endl; }
void Fun(const int x){ cout const 左值引用 endl; }
void Fun(int x){ cout 右值引用 endl; }
void Fun(const int x){ cout const 右值引用 endl; }// 模板中的不代表右值引用而是万能引用其既能接收左值又能接收右值。
// 模板的万能引用只是提供了能够接收同时接收左值引用和右值引用的能力
// 但是引用类型的唯一作用就是限制了接收的类型后续使用中都退化成了左值
// 我们希望能够在传递过程中保持它的左值或者右值的属性, 就需要用到完美转发
templatetypename T
void PerfectForward(T t)//万能引用
{Fun(t);//调用fun函数时并不清楚t对象的类型
}std::forward 完美转发在传参的过程中保留对象原生类型属性
func( forwardT(t) )//完美转发保持原有属性 新的类功能
其实我们的移动构造和移动赋值函数也会默认生成但是条件是类中没有实现析构函数 、拷贝构造、拷贝赋值重载就是都没有实现才满足条件对于默认生成的移动构造函数对于内置类 型成员会执行逐成员按字节拷贝自定义类型成员则需要看这个成员是否实现移动构造如果实现了就调用移动构造没有实现就调用拷贝构造。
强制生成默认函数的关键字default 禁止生成默认函数的关键字delete