[关闭]
@cxm-2016 2016-09-30T10:51:19.000000Z 字数 4582 阅读 2021

C++:类的构造函数和析构函数

c++ no

版本:2
作者:陈小默
说明:调整部分语序,修改部分别字

开发环境:Visual Studio 2010

构造函数


当我们需要在对象创建时初始化一些数据的时候,我们不可能提供一个普通的成员方法供使用者在对象创建后调用。因为如果使用者故意或者无意间忘记了调用该方法,就可能导致程序出现偏离预期的结果。为了防止这种情况的发生,C++中提供了一种特殊的成员函数--构造函数

声明和定义

构造函数与普通函数在声明与定义上别无二致,除了不需要声明返回值。
在声明中添加如下代码:

  1. Student(std::string name,int chScore,int enScore,int mathScore);

一般实现方式为:

  1. Student::Student(std::string name,int chScore=0,int enScore=0,int mathScore=0){
  2. cout<<"您好呀! "<<name<<",很高兴见到你!"<<endl;
  3. setName(name);
  4. setChScore(chScore);
  5. setEnScore(enScore);
  6. setMathScore(mathScore);
  7. }

引用参数与构造函数

当我们在类声明时声明了一个引用参数,而引用参数又要求必须初始化,且只能初始化一次。应该怎么办呢?这时我们需要一种新的成员初始化方式:

  1. std::string & name

比如我们的name属性是引用类型,那么我们需要在每一个构造方法实现的参数列表后面加上这样的初始化器

  1. Student::Student(std::string &n):name(n){
  2. ......
  3. }
  4. Student::Student(std::string &n,int chScore,int enScore,int mathScore):name(n){
  5. ......
  6. }

如果我们有多个引用需要初始化,这些构造器只需要使用逗号分割,并且除了引用类型,任何类型属性都可以使用这种方式初始化

  1. Class::Constructor(type *t1,type &t2,type t3...):t1(t1),t2(t2),t3(t3)...{}

使用构造函数定义对象

当我们声明过一个构造函数之后,我们可以有三种方式去调用它:
1,显式的调用

  1. Student jack = Student("jack",99,98,97);

2,隐式的调用

  1. Student jack("jack",99,98,97);

3,使用new运算符

  1. Student *jack = new Student("jack",99,98,97);

上述3中方法的关键区别在于内存分配:前两种方式的对象将会被在栈中创建,第3中方式对象将会在堆中被创建

默认构造函数

当我们没有显式的在声明中添加构造方法时,编译器会默认提供一个空参列表且没有任何行为的默认构造方法,就像下面这样

  1. Student(){}

但是当我们声明了自己的构造方法之后,编译器便不会再提供默认的构造方法了。
当我们想定义一个默认的构造器有两种方式:
1,给已有的构造函数的所有参数提供默认值

  1. Student(std::string name,int chScore=0,int enScore=0,int mathScore=0);

2,定义一个没有参数的函数重载

  1. Student();

由于每种类型的构造器只能声明一个,所以同时使用上述两种方式会报错。

注意:声明隐式方式调用空参构造方法不能使用括号,因为这时编译器会将这行代码解释为函数声明,比如:

  1. Student jack;//available
  2. Student jack();//invalidate

析构函数


析构函数是当对象被销毁时被系统调用的方法,一般用来清空内存。

声明和定义

析构函数的声明只用一种形式,就是~加上类名

  1. ~Student();

析构函数的实现就可以根据需求去定义

  1. Student::~Student(){
  2. cout<<"再见! "<<name<<"."<<endl;
  3. }

析构函数的执行时机

析构函数的执行与对象的创建位置相关
1,当对象被声明为全局变量,则当整个程序退出时,其析构函数才会被调用。
2,当对象被声明为自动存储类型(局部变量),当其生命周期结束,即其作用域执行完毕时,其析构函数会被调用。
3,当对象被声明为指针变量(使用new创建对象),当执行到delete方法时,其析构函数会被调用。
4,临时对象,这种对象在临时对象完成任务时自动调用其析构函数。实例演示中会专门介绍这种情况。

const成员函数

有如下代码:

  1. const Student jack("Jack",99,98,97);
  2. jack.show();//complier error

因为我们已经将jack对象声明为const类型,所以当我们在调用show()方法时,编译器并不知道其中有没有更改对象内容的方法。所以拒绝编译。如果我们非要这么做的话,可以采用以下方式对show()函数重新定义

  1. void show() const;

实例演示


1,在头文件student.h中声明类Student

  1. //student.h -- Student class interface
  2. //version 00
  3. #ifndef STUDENT_H_
  4. #define STUDENT_H_
  5. #include <string>
  6. class Student{ //class declaration
  7. private:
  8. std::string name;
  9. int ch;
  10. int en;
  11. int math;
  12. float average;
  13. void count(){
  14. average = (ch + en + math+0.0F)/3;
  15. }
  16. public:
  17. Student();
  18. Student(std::string name,int chScore,int enScore,int mathScore);
  19. ~Student();
  20. void setName(std::string name);
  21. void setChScore(int score);
  22. void setEnScore(int score);
  23. void setMathScore(int score);
  24. void show();
  25. };
  26. #endif

2,在文件student.cpp中实现类方法

  1. //student.cpp -- implementing the Student class
  2. //version 00
  3. #include "stdafx.h"
  4. #include "student.h"
  5. Student::Student(){
  6. cout<<"您好呀! 新同学,你叫什么名字?"<<endl;
  7. }
  8. Student::Student(std::string n,int chScore,int enScore,int mathScore):name(n),ch(chScore),en(enScore),math(mathScore){
  9. cout<<"您好呀! "<<name<<",很高兴见到你!"<<endl;
  10. }
  11. Student::~Student(){
  12. cout<<"再见! "<<name<<"."<<endl;
  13. }
  14. void Student::setChScore(int score){
  15. Student::ch = score;
  16. }
  17. void Student::setName(std::string n){
  18. Student::name = n;
  19. }
  20. void Student::setEnScore(int score){
  21. en = score;
  22. }
  23. void Student::setMathScore(int score){
  24. math = score;
  25. }
  26. void Student::show(){
  27. Student::count();
  28. ios_base::fmtflags orig = cout.setf(ios_base::fixed,ios_base::floatfield);
  29. std::streamsize prec = cout.precision(1);
  30. cout<<name<<" 同学的语文成绩为"<<ch<<"分,数学成绩为"<<math<<"分,英语成绩为"<<en<<"分,平均成绩"<<average<<"分"<<endl;
  31. cout.setf(orig,ios_base::floatfield);
  32. }

3,在mian方法中调用

  1. //visual studio 2010 --main program
  2. #include "stdafx.h"
  3. #include "student.h"
  4. int _tmain(int argc, _TCHAR* argv[]){
  5. Student *s1,s2,s3("Jack",90,80,70);
  6. s1 = new Student("Sam",90,95,100);
  7. s1->show();
  8. s2 = Student("Sue",85,90,95);
  9. s2.show();
  10. s3.show();
  11. s1 = new Student("Joe",91,92,93);
  12. s1->show();
  13. delete s1;
  14. return 0;
  15. }

根据上面的学习内容你猜猜运行结果是什么?

运行结果:
您好呀! 新同学,你叫什么名字?
您好呀! Jack,很高兴见到你!
您好呀! Sam,很高兴见到你!
Sam 同学的语文成绩为90分,数学成绩为100分,英语成绩为95分,平均成绩95.0分
您好呀! Sue,很高兴见到你!
再见! Sue.
Sue 同学的语文成绩为85分,数学成绩为95分,英语成绩为90分,平均成绩90.0分
Jack 同学的语文成绩为90分,数学成绩为70分,英语成绩为80分,平均成绩80.0分
您好呀! Joe,很高兴见到你!
Joe 同学的语文成绩为91分,数学成绩为93分,英语成绩为92分,平均成绩92.0分
再见! Joe.
再见! Jack.
再见! Sue.
请按任意键继续. . .

我们来一步一步的分析代码执行流程,

  1. Student *s1,s2,s3("Jack",90,80,70);

这一句代码声明了三个变量,并且定义了其中的两个!!! 这里声明了一个Student的指针类型s1,声明并定义了Student类型的变量s2,s3,后面的两种定义方式都是隐式的,只不过s2的定义如上述 默认构造函数 所述,容易使人产生这里仅仅是声明而不是定义的错觉。

  1. s2 = Student("Sue",85,90,95);

现在再来看这一句,这里产生了一个上述临时对象。这个临时对象会将其内部的值复制到s2的空间中,其后自动销毁。那么为什么不让s2的引用指向这个新建立的对象呢?这就要从变量内存上分析,由于我们的s2和s3是在栈中分配的内存,而栈又会随着方法的退出而销毁,就导致了栈中的对象是不稳定的。如果我们采用指向的方式,那么我们就不知道什么时间我们指向的栈帧就会撤销,这时我们再去访问我们的方法就会报错了。

  1. s1 = new Student("Joe",91,92,93);

由于s1是在堆中分配的内存,所以我们才能在new时直接传递引用,但是这种写法是一个不好的编程习惯。因为我们在给指针重新分配内存的时候并没有释放原空间,如果产生大量这种代码将会造成内存溢出

最后,为什么先释放的是s3,后释放的是32呢?是由于栈的FILO性质导致的。

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注