0238029

练习2.38:说明由decltype指定类型和由auto指定类型有何区别。请举出一个例子,decltype指定的类型与auto指定的类型一样;再举一个例子,decltype指定的类型与auto指定的类型不一样。

【出题思路】

auto和decltype是两种类型推断的方式,本题旨在考查二者的区别和联系。

【解答】

auto和decltype的区别主要有三个方面:

第一、auto类型说明符用编译器计算变量的初始值来推断其类型,而decltype虽然也让编译器分析表达式并得到它的类型,但是不实际计算表达式的值。

第二、编译器推断出来的auto类型有时候和初始值的类型并不完全一样,编译器会适当地改变结果类型使其更符合初始化规则。例如,auto 一般会忽略掉顶层const,而把底层const保留下来。与之相反,decltype会保留变量的顶层const。

第三、与auto不同,decltype的结果类型与表达式形式密切相关,如果变量名加上了一对括号,则得到的类型与不加括号时会有不同。如果decltype使用的是一个不加括号的变量,则得到的结果就是该变量的类型;如果给变量加上了一层或多层括号,则编译器将推断得到引用类型。

一个用以说明的示例如下所示:

#include <iostream>
#include <typeinfo>
int main()
{
    int a = 3;
    auto c1 = a;
    decltype(a) c2 = a;
    decltype((a)) c3 = a;
    const int d = 5;
    auto f1 = d;
    decltype(d) f2 = d;
    std::cout << typeid(c1).name()  << std::endl;
    std::cout << typeid(c2).name()  << std::endl;
    std::cout << typeid(c3).name()  << std::endl;
    std::cout << typeid(f1).name()  << std::endl;
    std::cout << typeid(f2).name()  << std::endl;
    c1 ++ ;
    c2 ++ ;
    c3++ ;
    f1 ++ ;
    //f2 ++ ;   //错误:f2是整型常量,不能执行自增操作
    std::cout<<" "<<c1<<" "<<c2<<" "<<c3<<" "<<f1<<" "<<f2<<std::endl;
    return 0;
}
/*
i
i
i
i
i
 4 4 4 6 5
*/

对于第一组类型推断来说,a是一个非常量整数,c1的推断结果是整数,c2的推断结果也是整数,c3的推断结果由于变量a额外加了一对括号所以是整数引用。c1、c2、c3依次执行自增操作,因为c3是变量a的别名,所以c3自增等同于a自增,最终a、cl、c2、c3的值都变为4。

对于第二组类型推断来说,d是一个常量整数,含有顶层const,使用auto推断类型自动忽略掉顶层const,因此f1的推断结果是整数;decltype则保留顶层const,所以f2的推断结果是整数常量。f1可以正常执行自增操作,而常量f2的值不能被改变,所以无法自增。

本页共48段,2252个字符,3518 Byte(字节)