C++设计类的注意事项

作者: Cloudox_ | 来源:发表于2017-12-13 09:55 被阅读24次

构造函数

如果没有声明构造函数,编译器会定义一个默认构造函数(无参数、无内容),让你可以不初始化来直接创建对象:

Star rigel;
Star pleiades[6];

但如果定义了某种形式的构造函数,编译器就不会帮你定义默认构造函数了,如果还是有上述情况的使用,那需要自己定义一个默认构造函数。最好提供一个显式的默认构造函数,保证不出错。

构造函数用来创建新对象,它是不能被派生类继承的,派生类需要定义自己的构造函数,并在初始化列表中调用基类的构造函数:

SubClass::SubClass(int a, int b):BaseClass(b) {
  // 派生类初始化部分
}

注意,初始化列表只能在构造函数上使用。

和普通构造函数一样,如果你没定义复制构造函数,编译器将提供一个,旦最好显式地自己定义一个,对于一些用new初始化的成员,自行用深复制来做复制,否则编译器提供的只是简单的浅复制,在删除时会出问题。下面这些情况会用到复制构造函数:

  • 将新的对象初始化为一个同类对象。
  • 按值将对象传递给函数。
  • 函数按值返回对象。
  • 编译器生成临时对象。

赋值操作符

要分清楚什么是赋值,什么是初始化,这是不同的:

Star sirius;
Star alpha = sirius;// 初始化
Star dogstar;
dogstar = sirius;// 赋值

最好也显示定义赋值操作符,如果你可能用到的话。大概如下:

Star & Star::operator=(const char &) {...}

注意这里传递的参数与是引用,因为基类引用可以指向派生类,而派生类引用不可指向基类,所以可以将派生类赋值给一个基类,而不能将一个基类赋值给一个派生类(毕竟缺少成员)。同理,如果要做到不同类之间的赋值(也包括基类赋值给派生类),要么做强制类型转换再赋值,要么定义一个特定参数的赋值操作函数。

赋值操作符也是不能被继承的,毕竟其特征标(参数列表)随类而异。

在定义派生类的赋值操作符重载函数时,要显式地在函数块中通过::来调用基类的赋值操作符,来操作基类的成员,毕竟派生类很多时候无法直接访问到基类成员,只能通过调用基类的公开方法来访问,而且也不能通过初始化列表的方式来调用:

SubClass & SubClass::operator=(const SubClass & sub) {
  if (this == &sub)
    return *this;
  BaseClass::operator=(sub);// 显示调用基类赋值操作符函数
  //注意这个函数的参数应该是基类引用,但是基类引用是可以指向子类的,它只会操作基类的成员
  ...// 操作派生类的成员
  return *this;
}

析构函数

一定要注意显式定义析构函数来释放构造函数使用new分配的所有内存。

基类的析构函数最好定义成虚函数(virtual),这样当释放一个基类指针指向的派生类时,也会自动先调用派生类的析构函数,然后才调用基类的析构函数,否则会只调用基类的析构函数,这样派生类用new初始化的成员将得不到释放。

按值与按引用传递对象

通常如果函数参数是对象,最好按引用来传递,一是为了提高效率,毕竟按值传递需要复制一个对象,这就要调用复制构造函数,用完了再调用析构函数,这很费时间,尤其是当类比较大的时候。而按引用传递则很快。另外,也由于C++支持用基类的引用指向派生类时,对于虚函数会调用其真实类型的函数,这保证了灵活的使用。只是要注意如果在函数中不修改对象,最好用const修饰对象参数,避免修改。

当把对象作为返回值时,如果是传递的原始对象引用,那么要返回对象的引用,保证是传递的同一个对象,比如重载<<操作符时,就要传递同一个cout对象,因此必须返回引用,按引用返回也可以节省时间。但是如果返回的是函数中临时创建的新对象,那就只能按值返回,毕竟函数结束后这个新对象就会被析构了,必须复制一个对象来传递回去。

私有成员与保护成员

用private修饰的为私有成员,只有类对象自己可以访问,派生类也不可以。用protected修饰的为保护成员,类对象自己可以访问,派生类也可以访问,外部类不能访问。用public修饰的就都可以访问了。

将基类成员设为private的可以提高安全性,但是设为protected则可以简化代码,提高访问速度,这就按需索取啦。

虚函数

前面也提高过,用virtual修饰类的一个函数原型可以令其变成虚函数(虚方法)。只需要在原型时修饰,定义中不用再次修饰。

用virtual修饰的虚函数,在派生类中也会自动成为虚函数。虚函数的意义是当用基类的指针或引用指向对象时(不管指向的是基类对象还是派生类对象),调用虚函数会根据对象真实类型调用对应方法。如果没用virtual修饰,那用基类指针或引用去调用方法时只会调用基类的方法:

virtual void func();

BaseClass base = ...
SubClass sub = ...
base.func();// 调用基类的方法
sub.func();// 调用派生类的方法
BaseClass &ref = sub;
ref.func();// 调用派生类的方法,如果不用virtual修饰,则调用基类方法

当然,如果要能做到分开调用,在派生类中也要一模一样的定义一个方法(参数列表要一致),此时用virtual修饰与否都可以,毕竟基类已经修饰过了,其所有派生类和派生类的派生类的此方法都是虚方法。但最好还是用virtual修饰一下,比较明显。

所有要在派生类中重定义的方法都建议在基类中用virtual修饰,以防出错。

如果更近一步,在声明虚方法时后面加个const=0,这叫做纯虚方法:

virtual void func()const = 0;

这会让此类变成一个抽象基类,抽象基类的意思是它就是一个专门用作基类的,不能初始化它的对象出来,比如有一个类是“圆形类”,一个类是“椭圆形类”,为了方便可以定义一个“形状类”作为它们两个的抽象基类,持有一些比如圆心坐标等的共有成员和方法,但是你不能去创建一个“形状”对象来,没什么意义。这时就可以用到抽象基类了,也就是至少让一个方法使用上面的声明方式。


查看作者首页

相关文章

  • C++设计类的注意事项

    构造函数 如果没有声明构造函数,编译器会定义一个默认构造函数(无参数、无内容),让你可以不初始化来直接创建对象: ...

  • C++ — 类 & 对象超详解

    C++ 类 & 对象 C++ 在 C 语言的基础上增加了面向对象编程,C++ 支持面向对象程序设计。类是 C++ ...

  • C++零基础教程之类和对象初识

    C++ 类和对象 C++ 在 C 语言的基础上增加了面向对象编程,C++ 支持面向对象程序设计。类是 C++ 的核...

  • C++面向对象

    C++类和对象 C++ 在 C 语言的基础上增加了面向对象编程,C++ 支持面向对象程序设计。类是 C++ 的核心...

  • C++ 类 & 对象

    原文地址:C++ 类 & 对象 C++ 在 C 语言的基础上增加了面向对象编程,C++ 支持面向对象程序设计。类是...

  • 读《C++沉思录》有感

    关于类的设计:代理类 《C++沉思录》的原话是这样的 我们怎样才能设计一个C++容器,使它有能力包含类型不同而彼此...

  • NDK开发—C++面向对象编程(四)

    类 C++ 在 C 语言的基础上增加了面向对象编程,C++ 支持面向对象程序设计。类是 C++ 的核心特性,用户定...

  • CPP基础:面向对象编程

    面向对象编程 类 C++ 在 C 语言的基础上增加了面向对象编程,C++ 支持面向对象程序设计。类是 C++ 的核...

  • 2022-08-01# 面向对象编程

    类 C++ 在 C 语言的基础上增加了面向对象编程,C++ 支持面向对象程序设计。类是 C++ 的核心特性,用户定...

  • C++基础-(IO和文件操作)

    C++基础 IO和文件操作 标准输入输出的设备名有哪些 文件输入类的类名是:iostream 输出流注意事项计算顺...

网友评论

    本文标题:C++设计类的注意事项

    本文链接:https://www.haomeiwen.com/subject/emtaixtx.html