【你不知道】表达式中的隐式类型转换、无名对象作为函数实参
2010-04-09 17:07
288 查看
代码
1 int array[] = {1,2,3,4,5};
2 #define TOTAL_ELEMENTS (sizeof(array)/sizeof(array[0]))
3
4 #include <iostream>
5 using namespace std;
6
7 class Base{
8 public:
9 explicit Base(int i=0):ival(i)
10 {
11 cout << "Class Base's constructure is called" << endl;
12 }
13 Base(const Base& b)
14 {
15 cout << "Class Base's copy constructure is called" << endl;
16 }
17 public:
18 int ival;
19 };
20
21 void fun(Base b)
22 {
23 cout << "fun is called" << endl;
24 }
25
26 Base fn1()
27 {
28 Base b;
29 return b;
30 }
31
32 Base fn2()
33 {
34 return Base();
35 }
36
37 void main()
38 {
39 int val = 5;
40 fun(Base(val));
41 Base myb;
42 fun(myb);
43 cout << endl;
44
45 int d = -1;
46 if(d < (unsigned char)1)
47 cout << "int -1 < unsigned char 1" << endl;
48 if(d < (unsigned int)1)
49 cout << "int -1 < unsigned int 1" << endl;
50
51 if(d <= TOTAL_ELEMENTS)
52 cout << "int -1 < TOTAL_ELEMENTS" << endl;
53
54 cout << endl;
55
56 fn1(); cout << endl;
57 fn2(); cout << endl;
58
59 Base b1 = fn1(); cout << endl;
60 Base b2 = fn2(); cout << endl;
61
62 getchar();
63 }
运行结果:
注意40行,用一个无名对象作为实参,调用函数fun,但编译器用实参初始化形参的时候,并没有如用普通对象调用fun一样,调用拷贝构造函数。无名对象的初始化,也就是形参的初始化。
这个技术还可以用于函数的return语句。如:return Student("Lucy"); 或者 Student st("Lucy"); return st; 前者好于后者。(例子fn1、fn2函数的调用)
对于后者,st对象被创建,同时调用构造函数完成初始化,然后调用拷贝构造函数,把st拷贝到保存返回值的外部存储单元中。最后,st对象在函数结束时被销毁。对于前者,编译器可以直接把临时对象创建并初始化在外部存储单元中,省去了拷贝和析构函数的开销。
关于比较的说明:当一个表达式的操作数类型不同,就会发生转换。数据类型一般朝着浮点精度更高、占用内存空间更长的方向转化。当int -1和unsigned char 1 比较的时候,后者隐式转换为int类型的1,所以前者小于后者,关系成立。
当int -1 和unsigned int 1比叫大小,前者转换为unsigned int,但-1转化为unsigned int为很大很大的值。所以前者小于后者,关系不成立。这个解释同样适用于TOTAL_ELEMENTS句子,sizeof返回的是无符号整形。
1 int array[] = {1,2,3,4,5};
2 #define TOTAL_ELEMENTS (sizeof(array)/sizeof(array[0]))
3
4 #include <iostream>
5 using namespace std;
6
7 class Base{
8 public:
9 explicit Base(int i=0):ival(i)
10 {
11 cout << "Class Base's constructure is called" << endl;
12 }
13 Base(const Base& b)
14 {
15 cout << "Class Base's copy constructure is called" << endl;
16 }
17 public:
18 int ival;
19 };
20
21 void fun(Base b)
22 {
23 cout << "fun is called" << endl;
24 }
25
26 Base fn1()
27 {
28 Base b;
29 return b;
30 }
31
32 Base fn2()
33 {
34 return Base();
35 }
36
37 void main()
38 {
39 int val = 5;
40 fun(Base(val));
41 Base myb;
42 fun(myb);
43 cout << endl;
44
45 int d = -1;
46 if(d < (unsigned char)1)
47 cout << "int -1 < unsigned char 1" << endl;
48 if(d < (unsigned int)1)
49 cout << "int -1 < unsigned int 1" << endl;
50
51 if(d <= TOTAL_ELEMENTS)
52 cout << "int -1 < TOTAL_ELEMENTS" << endl;
53
54 cout << endl;
55
56 fn1(); cout << endl;
57 fn2(); cout << endl;
58
59 Base b1 = fn1(); cout << endl;
60 Base b2 = fn2(); cout << endl;
61
62 getchar();
63 }
运行结果:
注意40行,用一个无名对象作为实参,调用函数fun,但编译器用实参初始化形参的时候,并没有如用普通对象调用fun一样,调用拷贝构造函数。无名对象的初始化,也就是形参的初始化。
这个技术还可以用于函数的return语句。如:return Student("Lucy"); 或者 Student st("Lucy"); return st; 前者好于后者。(例子fn1、fn2函数的调用)
对于后者,st对象被创建,同时调用构造函数完成初始化,然后调用拷贝构造函数,把st拷贝到保存返回值的外部存储单元中。最后,st对象在函数结束时被销毁。对于前者,编译器可以直接把临时对象创建并初始化在外部存储单元中,省去了拷贝和析构函数的开销。
关于比较的说明:当一个表达式的操作数类型不同,就会发生转换。数据类型一般朝着浮点精度更高、占用内存空间更长的方向转化。当int -1和unsigned char 1 比较的时候,后者隐式转换为int类型的1,所以前者小于后者,关系成立。
当int -1 和unsigned int 1比叫大小,前者转换为unsigned int,但-1转化为unsigned int为很大很大的值。所以前者小于后者,关系不成立。这个解释同样适用于TOTAL_ELEMENTS句子,sizeof返回的是无符号整形。
相关文章推荐
- B继承自A,A指针无法隐式转换为B指针,函数参数只管指针类型,与实际指向对象无关
- B继承自A,A指针无法隐式转换为B指针,函数参数只管指针类型,与实际指向对象无关
- 数据类型,隐式转换以及json,对象,引用类型,预解析 视频教程
- 解决springmvc关于前台日期作为实体类对象参数类型转换错误的问题
- (3)JavaScript基础(基本语法:变量与数据类型、类型转换、运算符、流程控制、函数、对象、自定义对象、原型)
- 关于C++中函数类型的隐式转换
- 隐式类型转换与非成员函数(effective C++ 条款24&46)
- C++ 对象构造, 拷贝, 赋值和隐式类型转换总结
- 无法确定条件表达式的类型,因为“<null>”和“System.DateTime”之间没有隐式转换 解决办法
- 无法确定表达式的类型,因为<null>和int之间没有隐式转换
- 函数参数的自动隐式类类型转换只能一次
- C语言表达式中的类型隐式转换
- javascript——对象的概念——函数 2 (内建函数与类型转换)
- 工作总结 无法确定条件表达式的类型,因为“<null>”和“System.DateTime”之间没有隐式转换 解决办法 object——Nullable<T> (可空类型)
- 三目表达式的隐式类型转换(转载)
- 对象类型转换,explicit,显式和隐式
- C++ Primer学习笔记——$14 操作符重载、函数对象及类类型转换
- com.microsoft.sqlserver.jdbc.SQLServerException: 不允许从数据类型 varbinary 到 date 的隐式转换。请使用 CONVERT 函数来运行此
- SQL中一些不经意隐式类型转换或者函数使用导致索引失效问题
- 不允许从数据类型 nvarchar 到 varbinary 的隐式转换。请使用 CONVERT 函数来运行此查询