欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

区分C++中的&和&&

程序员文章站 2022-06-26 12:33:26
c++的&和&&是两个常用且容易让人混淆迷惑的运算符,其中&符号有三种用途,&&有两种。本文对这两个运算符的用途做简要总结。&的用途第一种用途:...

c++的&和&&是两个常用且容易让人混淆迷惑的运算符,其中&符号有三种用途,&&有两种。本文对这两个运算符的用途做简要总结。

&的用途

第一种用途:位运算中的“与”(and)。位运算十分高效,数据分片时常会用到,例如网络数据报头、ip地址段、utf-8编码等。

第二种用途:取地址。这个功能在c中比较常见,比如取函数地址、变量地址。用法示例:

int b = 10;
int *a = &b; // a指针指向b的存储地址

// 声明函数
int add(int a, int b) {return a + b}
// 声明函数指针类型
int (*functionptr)(int, int);
// 声明和初始化指针, &取函数的地址
functionptr addptr = &add;

第三种用途:引用。这个功能是c++的补充,常用在函数传参(c中一般用指针)、临时变量引用等。用法示例:

// 声明v1和v2两个参数引用传递
template <typename t>
t product(const std::vector<t>& v1, const std::vector<t>& v2) {...}

std::vector<std::vector<int> > vecs(10);
// 引用取出数组里的值,避免复制
auto& vec = vecs[0];

可以看到,三种用法都很基础,使用上很简单。

&&的用途

第一种用途:“与”(and)逻辑运算符。做条件判断时,&&常用来连接多个条件。

第二种用途:右值引用,这个功能自c++11起才可用。移动语义是c++11新增的重要功能,其重点是对右值的操作。右值可以看作程序运行中的临时结果,右值引用可以避免复制提高效率。&&用法示例:

#include <iostream>
struct foo {
 ~foo() {std::cout << "destruction" << std::endl;}
};

foo foofactory() {
 return foo();
}

int main() {
 std::cout << "before copy constructor..." << std::endl;
 foo foo1 = foofactory();
 std::cout << "after copy constructor..." << std::endl << std::endl;
 // 引用右值,避免生成新对象
 foo&& foo2 = foofactory();
 std::cout << "life time ends!" << std::endl << std::endl;

 return 0;
}

clang编译器(加-fno-elide-constructors选项)编译上述代码, 运行结果如下:

before copy constructor...
destruction
destruction
after copy constructor...

destruction
life time ends!

destruction
destruction

从输出结果看,第二种写法少了一次destruction输出。这意味着通过右值引用(&&),foo2直接引用foofactory返回的对象,避免了对象复制。

注意: 由于“复制省略(copy elision)”优化技术,一些编译器输出可能与上述不同。例如g++编译器会略过临时对象复制以优化性能。g++ 4.8.5编译程序输出结果如下:

before copy constructor...
after copy constructor...

life time ends!

destruction
destruction

clang编译的结果对比,省去了三次临时对象的生成、析构。

参考

以上就是区分c++的&和&&的详细内容,更多关于c++ &和&&的资料请关注其它相关文章!

相关标签: c++ & &&