C++语法学习笔记二十五: 万能引用universal reference

实例代码

//万能引用 universal reference


#include <iostream>
#include <string>
#include <vector>

using namespace std;

void func(const int& a) { // a的类型: const int &

}

//上面函数改造成模板函数
template<typename T>
void func1(const T& a){

}

void myfunc(int &&tmprv) { //参数tmprv 是右值引用类型
	cout << tmprv << endl;
	return;
}

//将上面的 myfunc 函数改造下,
//改造后
//(1) tmprv 既能接受左值,又能接受右值了
//(2) tmprv 的类型是T&&
//
template<typename T> //模板函数
void myfunc1(T && tmprv) { //&&是属于tmprv 类型的一部分, &&和T类型 没有关系
	//T&&或者说tmprv 才是万能引用, 而单独的T 不是万能引用
	//如果我传递进来一个整形的左值, tmprv 就应该被推断成int&类型
	//如果我传递进来一个整形的右值,tmprv 它就应该被推断成int&&类型
	cout << tmprv << endl;
	return;
}

template<typename T>
void funcpd(std::vector<T>&& param) { //右值引用。 非万能引用

}

template<typename T> //模板函数
void myfunc2(const T && tmprv) {
	cout << tmprv << endl;
	return;
}


template<typename T> //模板类
class myTestc {

public:
	void testfunc(T&& x) { //这个x是个右值引用,不是万能引用, 因为在实例化后变成如: testfunc(int&& x) ,非testfunc(T&& x)

	} //T

	template<typename T2>
	void testfunc1(T2&& x){  //这是万能引用

	}

};


int main(int argc, const char * argv[]) {

	//一: 类型区别基本概念
	func1(10); //这里调用模板函数,T的类型是“int”, a的类型是“const int &” ;
	//(1) 调用函数模板时给的参数 10 会对T的类型产生影响。
	//(2) a的类型会对T的类型产生影响


	//下面的万能引用就说明了 a的类型会对T的类型产生影响
	//二: universal reference / 万能引用 / 未定义引用 基本认识
	//&& 表示右值引用。如下:
	int && rv = 100;

	myfunc(10); // ok
	int i = 100;
	//myfunc(i); //错误 右值引用不能接收(绑定)左值。无法将参数 i 从“int”转换为“int&&”

	myfunc1(i); //正确

	//万能引用语境:
	//a). 必须是函数模板
	//b). 必须是发生了模板类型推断并且函数模板参数的样式是:T&& (auto也存在万能引用的概念)
	//T&& : 就是万能引用


	//万能引用和右值引用的区别:
	//a). 右值引用得传递右值,否则编译器报错。
	//b). 万能引用作函数形参时,我们可以传递给它左值,也可以传递给它右值。
	//如果我们传递进去了左值,那么这个万能引用就变成了左值引用。
	//如果我们传递进去了右值,那么这个万能引用就变成了右值引用。

	//判断题:
	//(1). void func(int &&param); //右值引用,因为没模板
	//(2). template<typename T> void func(T&& tmpvalue) {}; // 万能引用
	//(3). template<typename T> void funcpd(std::vector<T>&& param) {}; //右值引用。 非万能引用

	std::vector<int> ab = { 1 };
	//funcpd(ab); // 错误 因为是右值引用 不能接收左值
	funcpd(std::move(ab)); //正确  std::move() 把左值转换为右值

	//三: 万能引用资格的剥夺与辨认
	//(1) 剥夺 : const 修饰词会剥夺一个引用成为万能引用的资格。会被打回原型成为右值引用。
	myfunc2(10);
	// myfunc2(i); // 错误  myfunc2 退化成右值引用, 不能接收左值

	//(2) 辨认
	myTestc<int> mc;
	int b = 100;
	//mc.testfunc(b);// 错误 testfunc函数参数y为右值引用 不能接收左值
	mc.testfunc(std::move(b));//正确  std::move 把左值转换为右值

	mc.testfunc1(b); //万能引用 可以接收左值

	system("pause");
	return 0;
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值