隐式类型转换

总结自:隐式类型转换&算数运算符

定义:隐式类型转换是指使用了与表达式规定或当前语境不相符的类型时所进行的类型转换,但是要注意,可能会存在转换出现歧义,从而无法通过编译;一切带有explicit的转换,构造函数,发生的类型转换均不属于隐式转换.

概念总括:

标准转换序列:

一个标准转换顺序可选的为以下:

  • 左值到右值的转换
  • 数值提升或转换
  • 函数指针转换
  • 限定调整

用户定义转换:

非explicit的构造函数和转换函数(与直接初始化不同)

算数转换:

算数转换是指一个表达式,运算符是算数运算符之一,此时有别于隐式类型转换的转换规则.

转换顺序:

  • 标准转换序列
  • 用户定义转换
  • 标准转换序列

接下来将对上述定义部分进行逐一说明:

  • 在语境中的隐式转换:

由于某语句中的表达式需要某种表达式类型,此时就会发生,比如if,逻辑运算符,switch,delete等,要求被转换的表达式的类型遵循转换顺序.
<
值得注意的是:对于此处的用户定义转换,

在C++ 14前,必须拥有一个转换为指定类型(或转换后可继续隐式转换为指定类型)的非explicit的转换函数
在C++ 14及之后则放宽为只要存在有可转换为指定类型(或转换后可继续隐式转换为指定类型)的限定或引用的转换函数即可

>

  • 左值到右值的转换:

这里统一盖称为左值到右值的转换;实际上,只是发生在表达式的运算数类型和当前运算数类型不同时,就会发生该转换.

此处转换实际上细分了三大类,但是总是伴随着泛左值向纯右值的转换以及纯右值之间的转换.


细节上逐一介绍:

  • 泛左值向纯右值的值类型转换

    • 会将任何非函数,数组的泛左值转换为同类型的纯右值,并且若非类类型,则会移除其cv限定,并以原左值的值作为右值的值,并且该转换不会对以下表达式求值:

      • 不求值语境,为nullptr_t的类型(直接转化为nullptr),不进行ODR访问的表达式(此处关于ODR,会有另外的一篇文章做详细描述).
    • 当其为类类型时则视为:
      • 在C++ 17前,以当前左值为参数所进行的一个复制构造函数的调用,构造函数返回的值为一个临时对象,故为一个纯右值表答式
      • 在C++ 17后,改称,该结果对象以泛左值对象复制初始化
      • 并且有,泛左值对象中存在不确定值时,初始化后行为未定义,除非:
      • 该值存在有指向它的指针或存在引用
      • 该值位于静态或TLS
  • 数值提升或转换
  • 数值提升或转换就是将原操作数的数据类型转换为其他的数据类型(可能损失精度),或几乎不会损失精度的类型提升;具体细节如下:

    • 整数类型提升:只需要保证,目标类型的有效值范围是大于等于原类型的即可
    • 整数类型转换:
      • 目标为无符号数发生溢出时,模目标类型的2nn为表示目标类型的位数
      • 目标为有符号数若发生溢出通常是实现定义.
        自C++20起,此时的值将会变成与源值对2n同余的唯一目标类型值.
      • 源为bool时,true为1,false为0
      • 目标是bool时,零值,空指针为false,其余的均为true
      • 浮点数:浮点数在发生类型转换时,如果原值可表示,但是存在多种表示时,由实现定义,或可被精确表示时,值不变,否则未定义,若向整数转换,要求整数对浮点部分截断,无舍入,并且当无法表示时,行为未定义.
      • 指针, NULL可以转换为任何类型的指针类型,void的(要求同cv等级)指针类型可以作为同cv等级的指针转换的目标类型,在基类可访问或无歧义时,可以将派生类的指针转为为其基类的指针(CV等级相同)
        • 函数指针转换:
        • 函数指针实际上很简单,即任何非非静态成员成员函数的id(一个类型为T的函数左值)都可以转换为该函数类型的指针(一个指向改函数类型T的指针纯右值),为什么非静态成员函数不可以呢?因为任何访问非静态成员函数的表达式都是被归为纯右值的,其仅只能作为调用运算符的操作数.
        • 临时变量实质化:
        • 这个是一个纯右值到亡值的过程,即当一个纯右值出现在某些特定表达式时,会发生实质化,变为一个亡值(注,基本上,除了允许纯右值直接作为操作数的表达式之外(例如,所有的隐式类型转换均伴随左值到纯右值的转化为第一步,其后再发生类型转换的,此时类型改变,值类型任然为纯右值,允许作为表达式操作数),均要发生该转换),标准中有如下:

          • 成员访问,下标,这类成员访问的表达式
          • 作为弃值表达式的最终类型
          • 引用绑定到右值时
          • 特殊的:不求值表达式的操作数(sizeof,typeid)
        • 数组到指针的转换:该规则很简答,直接转换为指向数组第一个元素类型的指针即可
        • 限定调整:
        • 对于任何一个带有CV限定的类型T我们都可以做如下对应:
          CV0 P0 CV1 P1 CV2 P2 ... CVn - 1 Pn - 1 CVn U
          例一:如类型const char * * 此时做对应实际上有:
        const char * *
        CV2 U CV1P1 CV0P0

          实际上观察可以发现,从标识符位置起,根据声明查看顺序反向阅读即可得到对应关系,再比如: char * A [];

        char * []
        CV2 U CV1P1 CV0P0

          当然,此处的U可以做任何包含,即可以包含N组(至少留一组)CVn Pn - 1


          由此,上述例1的CV分解实际上U有两层,即U为char或U为const char *


          对于标准转换中的限定调整,其规则实际上很简单,一句话概括就是当且仅当目标的U和对应的CV层数相同且CV限定符一致,或当某级的cv限定符更多时,要求除0级外均有cv限定时才可以转换,例如:

        char ** p;
        const char ** p1 = p; //错误,二级存在更多cv限定,但是1级没有
        const char * const * p2 = p; //正确

        C++ 类型转换(conv.)的更多相关文章

        1. Util应用程序框架公共操作类(三):数据类型转换公共操作类(扩展篇)

          上一篇以TDD方式介绍了数据类型转换公共操作类的开发,并提供了单元测试和实现代码,本文将演示通过扩展方法来增强公共操作类,以便调用时更加简化. 下面以字符串转换为List<Guid>为例进 ...

        2. Util应用程序框架公共操作类(二):数据类型转换公共操作类(源码篇)

          上一篇介绍了数据类型转换的一些情况,可以看出,如果不进行封装,有可能导致比较混乱的代码.本文通过TDD方式把数据类型转换公共操作类开发出来,并提供源码下载. 我们在 应用程序框架实战十一:创建VS解决 ...

        3. 彻底理解c++的隐式类型转换

          隐式类型转换可以说是我们的老朋友了,在代码里我们或多或少都会依赖c++的隐式类型转换. 然而不幸的是隐式类型转换也是c++的一大坑点,稍不注意很容易写出各种奇妙的bug. 因此我想借着本文来梳理一遍c ...

        4. 为C# as 类型转换及Assembly.LoadFrom埋坑!

          背景: 不久前,我发布了一个调试工具:发布:.NET开发人员必备的可视化调试工具(你值的拥有) 效果是这样的: 之后,有小部分用户反映,工具用不了(没反应或有异常)~~~ 然后,建议小部分用户换个电脑 ...

        5. c# 基础 object ,new操作符,类型转换

          参考页面: http://www.yuanjiaocheng.net/webapi/config-webapi.html http://www.yuanjiaocheng.net/webapi/web ...

        6. Struts2日期类型转换

          针对日期类java.util.Date进行类型转换,要求客户端使用"yyyy-MM-dd","yyyy/MM/dd"中的任意一种输入,并以"yyyy- ...

        7. 【.NET深呼吸】基础:自定义类型转换

          照例,老周在开始吹牛之前,先讲讲小故事,这是朋友提出的建议,老TMD写技术有什么了不起的,人人都会写.后来老周想想,也确实,代码谁不会写,能写到有品位有感悟,就不容易做到.于是,老周接受了该朋友的建议 ...

        8. C++四种类型转换方式。

          类型转换有c风格的,当然还有c++风格的.c风格的转换的格式很简单(TYPE)EXPRESSION,但是c风格的类型转换有不少的缺点,有的时候用c风格的转换是不合适的,因为它可以在任意类型之间转换,比 ...

        9. struts2类型转换

          1. Struts2中的类型转换 我们知道通过HTTP提交到后台的数据,都是字符串的形式,而我们需要的数据类型当然不只字符串类型一种.所以,我们需要类型转换! 在Struts2中,类型转换的概念除了用 ...

        随机推荐

        1. [转]jquery的ajax交付时“加载中”提示的处理方法

          本文转自:http://www.educity.cn/wenda/77121.html jquery的ajax提交时“加载中”提示的处理方法    方法1:使用ajaxStart方法定义一个全局的“加 ...

        2. (转)Rsync 排错案例解析

          Rsync 排错案例解析 原文:http://blog.51cto.com/irow10/1827306 错误一. 执行计划任务的备份脚本后没有看到备份的文件 1.首先查看crontab日志是否执行文 ...

        3. 修改linux文件权限命令

          修改linux文件权限命令:chmod Linux系统中的每个文件和目录都有访问许可权限,用它来确定谁可以通过何种方式对文件和目录进行访问和操作. 文件或目录的访问权限分为只读,只写和可执行三种.以文 ...

        4. Kudu的概念术语

          不多说,直接上干货! Columnar Data Store(列式数据存储) Kudu 是一个 columnar data store(列式数据存储).列式数据存储在强类型列中.由于几个原因,通过适当 ...

        5. rake 任务参数传递问题解决

          原文 :  https://robots.thoughtbot.com/how-to-use-arguments-in-a-rake-task namespace :tweets do desc 'S ...

        6. 【转载】CSS3 常用四个动画(旋转、放大、旋转放大、移动)

          http://blog.csdn.net/fungleo/article/details/49848905

        7. Swift-取消传统For循环

          1.取消传统的For循环 传统的for,在swift 3.0 被取消 i++/++i在swift 3.0 被取消 i += 1代替 for var i = 0;i<10;i +=1 { } 2. ...

        8. 信息无缝滚动效果marquee

          横向滚动.纵向滚动 1. 解决滚动的空白 向左向右滚动的话,可以根据父级定位left,每次加或者减可以使物体向左或右运动,用top也可以实现向上或向下运动 上下滚动实现无缝滚动1. innerHTML ...

        9. mybatis由JDBC的演化过程分析

          我们知道,mybatis是对JDBC的封装,那么他是如何演变过来的呢? 摘自传智传媒Java培训资料 关于mybatis的演化原理,我们先看看我们最熟悉也是最基础的通过JDBC查询数据库数据,一般需要 ...

        10. IPV4基本知识介绍

          转自华为官网 1.1  介绍 定义 IPv4(Internet Protocol Version 4)协议族是TCP/IP协议族中最为核心的协议族.它工作在TCP/IP协议栈的网络层,该层与OSI参考 ...