1 // main.cpp
2
3 #include <conio.h>
4 #include <iostream>
5 using namespace std;
6
7 class Ptr
8 {
9 public:
10 int* a;
11 Ptr() // 声明不初始化用构造函数
12 {
13 a = 0;
14 }
15 Ptr(int* p) // 初始化用构造函数
16 {
17 this->a = p;
18 }
19 Ptr(Ptr& p)
20 {
21 this->a = p.a;
22 }
23 void operator=(int* p) // 赋值用等号操作符重载
24 {
25 this->a = p;
26 }
27 void operator=(Ptr& p)
28 {
29 this->a = p.a;
30 }
31 operator int*() // 隐式类型转换函数
32 {
33 return a;
34 }
35 };
36
37 void main()
38 {
39 int i = 10;
40 Ptr p0; // 调用Ptr()
41 Ptr p1; // 调用Ptr()
42 Ptr p2 = &i; // 调用Ptr(int* p)
43 Ptr p3 = p1; // 调用Ptr(Ptr& p)
44 p0 = &i; // 调用void operator=(int* p)
45 p1 = p0; // 调用void operator=(Ptr& p)
46 getch();
47 }
从这个短小而恶心的程序中,我们可以看到以下几个容易被忽视的知识点:
1. 要想只声明变量而不进行初始化,最好追加无参数构造函数。
(如果程序中,已经添加了其他形式的构造函数,就必须添加无参数构造函数。)
2. C++中,变量的初始化和赋值两种操作是具有明显区别的。
在变量声明的时候,直接进行赋值的操作算作初始化。例如:
Ptr p = &i;
此时,C++将自动调用参数匹配的构造函数或拷贝构造函数。
在变量声明之后,再对该变量进行赋值操作算作赋值操作。例如:
Ptr p;
p = &i;
此时,C++将自动调用参数匹配的等号(=)操作符重载。
3. 如果想把一个类的对象转换成其他不相干的类型的对象需要追加隐式类型转换函数。
通常来说,如果无参数构造函数,初始化用构造函数和等号操作符重载中任意两个存在追加的必要,
那么,剩下的一个也必须被追加。