C++中隐式类型转换学习笔记

 更新时间:2020年02月25日 09:31:30   作者:dnsir  
在本篇文章里小编给大家整理的是一篇关于C++中隐式类型转换学习笔记内容,有兴趣的跟着小编来学习下吧。

1 operator隐式类型转换

1.1 std::ref源码中reference_wrapper隐式类型转换

在std::ref的实现中有如下一段代码:

template<typename _Tp>
 class reference_wrapper
 : public _Reference_wrapper_base<typename remove_cv<_Tp>::type>
 {
  _Tp* _M_data;

 public:
  typedef _Tp type;

  reference_wrapper(_Tp& __indata) noexcept
  : _M_data(std::__addressof(__indata))
  { }

  reference_wrapper(_Tp&&) = delete;

  reference_wrapper(const reference_wrapper&) = default;

  reference_wrapper&
  operator=(const reference_wrapper&) = default;
  //operator的隐式类型转换
  operator _Tp&() const noexcept
  { return this->get(); }

  _Tp&
  get() const noexcept
  { return *_M_data; }

  template<typename... _Args>
 typename result_of<_Tp&(_Args&&...)>::type
 operator()(_Args&&... __args) const
 {
  return __invoke(get(), std::forward<_Args>(__args)...);
 }
 };

注意看operator操作符重载:

operator _Tp&() const noexcept
  { return this->get(); }

就是用于类型转换。

1.2 简单的例子-实现一个class转为int的示例

#include <iostream>
/*
 *
 * c++ operator的隐式类型转换
 * 参见std::ref的实现
 */ 
void f(int a)
{
 std::cout << "a = " << a << std::endl;
}
class A{
 public:
 A(int a):num(a){}
 ~A() {}

 operator int()
 {
  return num;
 }
 int num;
};

int main()
{
 A a(1);
 std::cout << a + 1 << std::endl;
 f(a);
 return 0;
}

当然除了通过operator实现隐式类型转换,c++中还可以通过构造函数实现。

2 构造函数实现隐式类型转换

在c++ primer一书中提到

可以用单个实参来调用的构造函数定义了从形参类型到该类类型的一个转换

看如下示例:

#include <iostream>
/*
 *
 * c++ 构造的隐式类型转换
 * 参见std::ref的实现
 */
class B{
 public:
 B(int a):num(a){}
 ~B() {}

 int num;
};

class A{
 public:
 A(int a):num(a){}
 A(B b):num(b.num){}
 ~A() {}

 int fun(A a)
 {
  std::cout << num + a.num << std::endl;
 }

 int num;
};

int main()
{
 B b(1);
 A a(2);
 //通过构造函数实现了隐式类型转换
 a.fun(b); //输出结果为3
 return 0;
}

特别需要注意的是单个实参,构造函数才会有隐式转换,一个条件不满足都是不行。

3 使用explicit关键字避免构造函数隐式转换

有些时候我们并不希望发生隐式转换,不期望的隐式转换可能出现意外的结果,explicit关键词可以禁止之类隐式转换,将上述class A的构造函数改为如下

class A{
 public:
 A(int a):num(a){}
 explicit A(B b):num(b.num){}
 ~A() {}

 int fun(A a)
 {
  std::cout << num + a.num << std::endl;
 }

 int num;
};

再次运行程序出现提示:

op2.cpp: In function ‘int main()':
op2.cpp:29:12: error: no matching function for call to ‘A::fun(B&)'
a.fun(b);
^
op2.cpp:16:9: note: candidate: int A::fun(A)
int fun(A a)
^~~
op2.cpp:16:9: note: no known conversion for argument 1 from ‘B' to ‘A'

这个时候调用方式修改更改为:

int main()
{
 B b(1);
 A a(2);
 a.fun(A(b));
 return 0;
}

只能感叹C++语言的博大精深,这篇文章还只是对隐式转换的入门级总结。

参考:

《C++ Primer》隐式类类型转换学习整理

以上就是C++中隐式类型转换学习笔记的详细内容,更多关于C++中隐式类型转换的资料请关注脚本之家其它相关文章!

相关文章

  • C语言由浅入深讲解线程的定义

    C语言由浅入深讲解线程的定义

    这篇文章主要介绍了C语言中线程的基础知识,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-05-05
  • C++ 数据结构 堆排序的实现

    C++ 数据结构 堆排序的实现

    这篇文章主要介绍了C++ 数据结构 堆排序的实现的相关资料,需要的朋友可以参考下
    2017-06-06
  • C++ 基于BFS算法的走迷宫自动寻路的实现

    C++ 基于BFS算法的走迷宫自动寻路的实现

    这篇文章主要为大家介绍了C++ 基于BFS算法实现走迷宫自动寻路,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-11-11
  • 详解C++中构造函数,拷贝构造函数和赋值函数的区别和实现

    详解C++中构造函数,拷贝构造函数和赋值函数的区别和实现

    这篇文章主要介绍了C++中构造函数,拷贝构造函数和赋值函数的区别和实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-03-03
  • C语言菜鸟基础教程之判断

    C语言菜鸟基础教程之判断

    C语言判断结构要求程序员指定一个或多个要评估或测试的条件,以及条件为真时要执行的语句(必需的)和条件为假时要执行的语句(可选的)
    2017-10-10
  • 关于C++内部类的介绍与使用示例

    关于C++内部类的介绍与使用示例

    今天小编就为大家分享一篇关于关于C++内部类的介绍与使用示例,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-12-12
  • C++中回调函数及函数指针的实例详解

    C++中回调函数及函数指针的实例详解

    这篇文章主要介绍了C++中回调函数及函数指针的实例详解的相关资料,希望通过本文能帮助到大家,让大家理解掌握这部分内容,需要的朋友可以参考下
    2017-10-10
  • Qt音视频开发之实现ffmpeg视频旋转显示

    Qt音视频开发之实现ffmpeg视频旋转显示

    这篇文章主要为大家详细介绍了在Qt音视频开发中如何利用ffmpeg实现视频旋转显示,文中的实现步骤讲讲清晰,感兴趣的小伙伴可以了解一下
    2023-03-03
  • C语言实现自行车存放管理系统

    C语言实现自行车存放管理系统

    这篇文章主要为大家详细介绍了C语言实现自行车存放管理系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-08-08
  • C++简明讲解缺省参数与函数重载的用法

    C++简明讲解缺省参数与函数重载的用法

    所谓缺省参数,顾名思义,就是在声明函数的某个参数的时候为之指定一个默认值,在调用该函数的时候如果采用该默认值,你就无须指定该参数。C++ 允许多个函数拥有相同的名字,只要它们的参数列表不同就可以,这就是函数的重载,借助重载,一个函数名可以有多种用途
    2022-06-06

最新评论