第8课 列表初始化(3)_防止类型收窄、explicit关键字
1. 防止类型收窄
(1)类型收窄:指的是导致数据内容发生变化或者精度丢失的隐式类型转换。
(2)类型收窄的几种情况:
①从浮点数隐式转换为整型数,如int i=2.2;
②从高精度浮点数隐式转换为低精度浮点数。如从long double转double或float。
③从整型数隐式转换为浮点数,并且超出浮点数的表示范围,如float=(unsigned long Long)-1,注意这表示将-1先强制转换为unsigned long long,再隐式转换为float。
④从整型数隐式转换为长度较短的整型数,并且超出其表示范围。如char x=65536。
(3)在C++98/03中,类型收窄编译器不会报错,而C++11中可以通过列表初始化来检查及防止类型收窄。
【编程实验】防止类型收窄
#include <iostream>
#include <vector>
using namespace std; int main()
{
int x1(5.3); //ok,x1=5
int x2 = 5.3; //ok,x2=5
int x3 {5.3}; //error:narrowing,gcc给的是warning
int x4 = {5.3};//同上
char c1{}; //ok
char c2{};//error:narrowing,gcc给的是warning
std::vector<int> v1{,,,,}; //ok
std::vector<int> v2{,,,,5.6}; //error:narrowing,gcc给的是warning float fa = 1e40; //ok,double->float
float fb = {1e40};//error,double->float,超过float能表示的范围
float fc = (unsigned long long)-; //将-1(0xFFFFFFFF)强转,再隐式转成float
float fd = {(unsigned long long)-}; //error
float fe = (unsigned long long); //将1强转,再隐式转成float
float ff = {(unsigned long long)}; //ok const int x = , y = ; //注意x,y被const修饰
char c = x; //ok
char d = {x};//error
char e = y; //ok
char f = {y};//ok。如果y为int型,则该行会出错。但由于加了const并且值为1。编译器
//认为这样的转换是安全的。 return ;
}
/*
*******************gcc上的测试结果*********************
e:\Study\C++11\8>g++ -std=c++11 test1.cpp
test1.cpp: In function 'int main()':
test1.cpp:9:13: error: narrowing conversion of '5.2999999999999998e+0' from 'double' to 'int' inside { } [-Wnarrowing]
test1.cpp:10:15: error: narrowing conversion of '5.2999999999999998e+0' from 'double' to 'int' inside { } [-Wnarrowing]
test1.cpp:12:15: error: narrowing conversion of '99999' from 'int' to 'char' inside { } [-Wnarrowing]
test1.cpp:14:33: error: narrowing conversion of '5.5999999999999996e+0' from 'double' to 'int' inside { } [-Wnarrowing]
test1.cpp:17:18: error: narrowing conversion of '1.0e+40' from 'double' to 'float' inside { } [-Wnarrowing]
test1.cpp:19:36: error: narrowing conversion of '18446744073709551615ull' from 'long long unsigned int' to 'float' inside { } [-Wnarrowing]
test1.cpp:24:11: warning: overflow in implicit constant conversion [-Woverflow]
test1.cpp:25:13: error: narrowing conversion of '1024' from 'int' to 'char' inside { } [-Wnarrowing] *******************vc2015上的测试结果*********************
test1.cpp(9): error C2397: 从“double”转换到“int”需要收缩转换
test1.cpp(10): error C2397: 从“double”转换到“int”需要收缩转换
test1.cpp(12): error C2397: 从“int”转换到“char”需要收缩转换
test1.cpp(14): error C2398: 元素“5”: 从“double”转换到“int”需要收缩转换
test1.cpp(17): error C2397: 从“double”转换到“float”需要收缩转换
test1.cpp(19): error C2397: 从“unsigned __int64”转换到“float”需要收缩转换
test1.cpp(25): error C2397: 从“const int”转换到“char”需要收缩转换
*/
2. explicit关键字
(1)explicit用于阻止编译器的隐式转换,一般用于修饰构造函数。
(2)C++98/03由于不能使用{}列表初始化,即隐式转换只发生在调用带一个参数的构造函数中。但C++11允许用{}列表初始化对象,这可能会隐式调用带多个参数的构造函数。
(3)C++11中,explicit可用于修饰带多个参数的构造函数以防止隐式转换。(注意C++98/03中explicit被用于修饰只带一个参数的构造函数,如果修饰带多个参数的构造函数则无效)
【编程实验】explicit修饰带多参的构造函数
#include <iostream>
using namespace std; class Test
{
public:
Test(int a, int b)
{
cout <<"Test(int a, int b)" << endl;
} Test(initializer_list<int>)
{
cout <<"Test(initializer_list<int>)" << endl;
} explicit Test(int a, int b, int c)
{
cout <<"explicit Test(int a, int b, int c)" << endl;
}
}; void func(const Test& t)
{ } struct Complex
{
int real, imag; //explicit
Complex(int re, int im = ):real(re), imag(im){} Complex operator+(const Complex& x)
{
return Complex((real+x.real),(imag+x.imag));
}
}; int main()
{
Complex c1(, );
Complex c2 = c1 + ;//会试图将5转换为Complex类型,所以会隐式调用构造
//函数。为了阻止这种行为,可以在构造函数前加explicit //1.由于Test带有一个initializer_list参数的构造函数,因此,下列
//用{}初始化的对象都会直接调用该构造函数。
//2.如果注释掉上述构造函数,则编译器会将{...}分解并传给相应的
//构造函数,如果找不到相应的带多参的构造函数,则直接报错。
Test t1(, ); //Test(int a, int b)
Test t2{, }; //Test(initializer_list<int>)
Test t3{, , }; //Test(initializer_list<int>)
Test t4 = {, }; //Test(initializer_list<int>)
Test t5 = {, , };//Test(initializer_list<int>)
Test t6(, , ); //ok,显式调用explicit Test(int a, int b, int c) func({, }); //Test(initializer_list<int>)
func({, , }); //Test(initializer_list<int>)
func(Test{, }); //Test(initializer_list<int>)
func(Test{, , }); //Test(initializer_list<int>) Test t11{, , , }; //Test(initializer_list<int>)
Test t12 = {, , , };//Test(initializer_list<int>)
Test t13 {}; //Test(initializer_list<int>) return ;
}
/*测试结果:
**************************不注释Test中带有一个initializer_list参数的构造函数***************************
e:\Study\C++11\8>g++ -std=c++11 test2.cpp
e:\Study\C++11\8>a.exe
Test(int a, int b)
Test(initializer_list<int>)
Test(initializer_list<int>)
Test(initializer_list<int>)
Test(initializer_list<int>)
explicit Test(int a, int b, int c)
Test(initializer_list<int>)
Test(initializer_list<int>)
Test(initializer_list<int>)
Test(initializer_list<int>)
Test(initializer_list<int>)
Test(initializer_list<int>)
Test(initializer_list<int>) **************************不注释Test中带有一个initializer_list参数的构造函数***************************
e:\Study\C++11\8>g++ -std=c++11 test2.cpp -fno-elide-constructors
test2.cpp: In function 'int main()':
test2.cpp:55:22: error: converting to 'Test' from initializer list would use explicit constructor 'Test::Test(int, int, int)'
Test t5 = {77, 5, 42}; test2.cpp:59:18: error: converting to 'const Test' from initializer list would use explicit constructor 'Test::Test(int, int, int)'
func({47, 11, 3}); test2.cpp:63:25: error: no matching function for call to 'Test::Test(<brace-enclosed initializer list>)'
Test t11{77, 5, 42, 500}; test2.cpp:64:28: error: could not convert '{77, 5, 42, 500}' from '<brace-enclosed initializer list>' to 'Test'
Test t12 = {77, 5, 42, 500}; test2.cpp:65:14: error: no matching function for call to 'Test::Test(<brace-enclosed initializer list>)'
Test t13 {10};
*/
第8课 列表初始化(3)_防止类型收窄、explicit关键字的更多相关文章
- 第7课 列表初始化(2)_分析initializer_list<T>的实现
1. 初始化列表的实现 (1)当编译器看到{t1,t2…tn}时便会生成一个initializer_list<T>对象(其中的T为元素的类型),它关联到一个array<T,n> ...
- C++11 列表初始化
在我们实际编程中,我们经常会碰到变量初始化的问题,对于不同的变量初始化的手段多种多样,比如说对于一个数组我们可以使用 int arr[] = {1,2,3}的方式初始化,又比如对于一个简单的结构体: ...
- c++11——列表初始化
1. 使用列表初始化 在c++98/03中,对象的初始化方法有很多种,例如 int ar[3] = {1,2,3}; int arr[] = {1,2,3}; //普通数组 struct A{ int ...
- 列表初始化 分析initializer_list<T>的实现
列表初始化(1)_统一初始化 1. 统一初始化(Uniform Initialization) (1)在C++11之前,很多程序员特别是初学者对如何初始化一个变量或对象的问题很容易出现困惑.因为可以用 ...
- C++ Union妙用(将列表初始化用于数组元素)
Union是个不被注意的关键字,意为联合体,这是个诡异的名字.若不是为了继承C语言,它也不会出现在C++中(虽说,union在C++中得到了扩充,完成了接近类的功能).它的作用主要是节省内存空间,在嵌 ...
- initializer_list 列表初始化
initializer_list 列表初始化 用花括号初始化器列表初始化一个对象,其中对应构造函数接受一个 std::initializer_list 参数. #include <iostrea ...
- 默认初始化&拷贝初始化&直接初始化&值初始化&列表初始化
一.各种初始化的形式 /* 定义变量形式一:不指定初始值 */ int a; // 默认初始化 /* 定义变量形式二:指定初始值 */ int b = 1; // 拷贝初始化 int b(1); // ...
- C++结构体成员列表初始化
C++关于struct和class的区别,可以看上一篇文章:c ++ class和struct[转] 结构体成员列表初始化,来个例子: #include <iostream> #inclu ...
- C++新标准:列表初始化
一.列表初始化意义 C++新标准为vector提供了一种新的初始化方式:列表初始化.适用于知道多个成员具体值的情况. 二.列表初始化用法 /*1.空vector<int>*/ vector ...
随机推荐
- 比较两个ranges(equal,mismatch,lexicographical_compare)
euqal 比较两个序列是否相等,相等返回true,不相等返回false //版本一:调用重载operator==比较元素 template <class InputIterator1,clas ...
- day5 大纲
01 昨日内容回顾 list: 增: append insert(index,object) extend() 迭代着追加 删: pop(默认删除最后一个)按照索引去删除 有返回值 remove 按照 ...
- 我发起并创立了一个 VMBC 的 子项目 D#
大家好, 我发起并创立了一个 VMBC 的 子项目 D# . 有关 VMBC , 请参考 <我发起了一个 用 C 语言 作为 中间语言 的 编译器 项目 VMBC> https ...
- Revit api 创建族并加载到当前项目
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...
- Maven项目main和test文件夹说明
需要自己来手动调整项目目录, Maven项目通常划分为 main 和 test 两部分,main 中存放实际项目资源,test 存放测试项目资源,二者内部同时又划分为 source 和 resourc ...
- PHP扩展开发:第一个扩展
在上一篇文章<PHP扩展开发:安装PHP>我们已经将开发PHP扩展的PHP环境安装成功,那么接下来采用最简单直接的方式创建第一个扩展. 我们先假设业务场景,是需要有这么一个扩展,提供一个叫 ...
- ML: 聚类算法R包 - 模型聚类
模型聚类 mclust::Mclust RWeka::Cobweb mclust::Mclust EM算法也称为期望最大化算法,在是使用该算法聚类时,将数据集看作一个有隐形变量的概率模型,并实现模型最 ...
- 第一个NIOS II工程using Qsys-------Let Qsys Say Hello
1.新建工程 2.添加原理图文件 注:似乎Nios II工程都需要涉及到原理图编程. 3.进入Qsys进行内核设计 注:启动Qsys后,系统已经为内核默认添加了一个组件clk_0. 4.设置时钟名字和 ...
- 5分钟搭建 nginx +php --------------(LNMP)新手专用
5分钟搭建 nginx +php --------------(LNMP)新手专用 2014-11-14 16:48 88876人阅读 评论(2) 收藏 举报 版权声明:本文为博主原创文章,未经博主允 ...
- C#性能优化总结
1. C#语言方面 1.1 垃圾回收 垃圾回收解放了手工管理对象的工作,提高了程序的健壮性,但副作用就是程序代码可能对于对象创建变得随意. 1.1.1 避免不必要的对象创建 由于垃圾回收的代价较高,所 ...