C++:类和对象(下)---对类和对象深入一些的理解

简介: C++:类和对象(下)---对类和对象深入一些的理解

构造函数

初始化列表

前面已然介绍过构造函数,但并未完全结束,构造函数有很多种写法,有带缺省参数的,有全缺省的,不带缺省参数的…但用前面的方法,都是对里面成员变量的一种赋值,也就是说,这是给类中每一个成员变量一个初始值

但这并不是初始化,这里要分清楚什么是初始化,什么是赋初值

简单来说,它们的一个区别就是初始化只能初始化一次,但是赋值可以多次赋值

因此构造函数中就引入了初始化列表的概念,初始化列表可以做到给类内的成员函数初始化,下面写一个具体的例子来表示

#include <iostream>
using namespace std;
class Date
{
public:
  Date(int year = 1900, int month = 1, int day = 1)
    : _year(year)
    , _month(month)
    , _day(day){}
  void Print()
  {
    cout << _year << "/" << _month << "/" << _day << endl;
  }
private:
  int _year;
  int _month;
  int _day;
};

上面就是对初始化列表的概述,通过这样的写法可以对参数进行初始化

这里就体现了初始化和赋值的区别:

  1. 如果采用的是初始化,那么每个成员变量在初始化列表中只能初始化一次
  2. 类内如果有引用成员函数,const成员变量和自定义成员且没有默认构造函数时,就要写在初始化列表的位置
  3. 尽量使用初始化列表进行初始化,不管是否用初始化列表,对于自定义类型的成员变量都会先使用初始化列表进行初始化
  4. 成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后次序无关

下面来对第二条进行解读,同时也能更好的理解上面的原理

  1. 有const成员变量
#include <iostream>
using namespace std;
class Date
{
public:
  //Date(int year = 1900, int month = 1, int day = 1)
  //  : _year(year)
  //  , _month(month)
  //  , _day(day){}
  Date(int year = 1900, int month = 1, int day = 1, int tmp = 1)
  {
    _year = year;
    _month = month;
    _day = day;
    _tmp = tmp;
  }
  void Print()
  {
    cout << _year << "/" << _month << "/" << _day << endl;
  }
private:
  int _year;
  int _month;
  int _day;
  const int _tmp;
};

这里的构造函数对吗?很显然是不能编译通过的,原因也很简单,这里的_tmp是const修饰的变量,怎么能给它赋值?解决方法有两个,第一个是可以给它在声明的时候就给它一个值,这是可以的,但使用初始化列表可以完美解决这个问题

#include <iostream>
using namespace std;
class Date
{
public:
  Date(int year = 1900, int month = 1, int day = 1, int tmp = 1)
    : _year(year)
    , _month(month)
    , _day(day)
    , _tmp(tmp) {}
  //Date(int year = 1900, int month = 1, int day = 1, int tmp = 1)
  //{
  //  _year = year;
  //  _month = month;
  //  _day = day;
  //  //_tmp = tmp;
  //}
  void Print()
  {
    cout << _year << "/" << _month << "/" << _day << endl;
    cout << "const tmp  " << _tmp << endl;;
  }
private:
  int _year;
  int _month;
  int _day;
  const int _tmp = 1;
};

这里也能更好理解赋值和初始化的区别,从const上就可以很好的体现出来

  1. 引用

还有一大使用场景就是在引用中可以体现,这里就不再写错误的示范了

class Date
{
public:
  Date(int year = 1900, int month = 1, int day = 1, int tmp = 1)
    : _year(year)
    , _month(month)
    , _day(day)
    , _tmp(tmp) {}
  //Date(int year = 1900, int month = 1, int day = 1, int tmp = 1)
  //{
  //  _year = year;
  //  _month = month;
  //  _day = day;
  //  //_tmp = tmp;
  //}
  void Print()
  {
    cout << _year << "/" << _month << "/" << _day << endl;
    cout << "const tmp  " << _tmp << endl;;
  }
private:
  int _year;
  int _month;
  int _day;
  int& _tmp;
};

explicit关键字

C++提供了关键字explicit,可以阻止不应该允许的经过转换构造函数进行的隐式转换的发生,声明为explicit的构造函数不能在隐式转换中使用

在了解explicit前,就必须先知道隐式转换是什么了

显式转换一定不陌生,就是强行转换,因此隐式转换就是编译器自动发生转换

比如说下面这样的例子:

int main()
{
  int i = 0;
  double d = 1.2;
  int& p = d;
  return 0;
}

这里编译是通不过的,原因是因为d是double类型因此编译不通过吗?其实不然

在编译器的处理中,会把d先生成一个临时变量,再进行赋值,而我们都知道临时变量是具有常性的,因此这里并不能把一个常属性的数交给引用来处理

因此这里只需要加上const,进行权限的缩小,就可以用常引用接纳常性变量了

const int& p = d;

其实,上述过程中把变量d从double类型转换到int类型就是一个隐式类型转换,我们并没有进行转换,但是编译器依旧自己转换了,并且生成了临时变量导赋值失败

而explicit的存在就是不能让隐式转换存在,因此我们不能把类型局限在int和double类型,要加入类的类型

class A
{
public:
  A(int a = 10) :_a(a) {}
private:
  int _a;
};
int main()
{
  A a;
  a = 2;
  return 0;
}

看看上面的代码发生了什么?把2赋给了一个类?如果你知道了隐式类型转换,那么就不难理解这段代码,这就是把2进行隐式类型转换,把它转换成了类A,里面成员变量_a的值是2,然后又进行了一次赋值

那么explicit就可以登场了,在初始化列表前面加上explicit:

class A
{
public:
  explicit A(int a = 10) :_a(a) {}
private:
  int _a;
};
int main()
{
  A a;
  a = 2;
  return 0;
}

此时,隐式转换就不存在了,因此这里的赋值就不复存在了,因为2在这里真的就是一个数字2

匿名对象

C++中引入了匿名对象的概念,简单来说就是没有名字的对象

class A
{
public:
  explicit A(int a = 10) :_a(a) {}
private:
  int _a;
};
int main()
{
  A(2);
  return 0;
}

这里看起来很奇怪,但这是可以通过的,C++内部是允许这样做的,它和有名对象的区别在于:

匿名对象的生命周期只在这一行,结束后就进行析构,而正常生成的对象需要在main函数结束后才会进行析构,普通生成的对象的生命周期是它的局部域

那匿名对象有什么用?

假设有这样的情景

class A
{
public:
  explicit A(int a = 10) :_a(a) {}
  void Print()
  {
    cout << "Print" << endl;
  }
private:
  int _a;
};
int main()
{
  A a1;
  a1.Print();
  A().Print(); // 匿名对象
  return 0;
}

假如这里我只需要调用类内的成员函数print,但如果正常来说我是需要创建一个对象,再通过对象去引用这个类内的成员函数,这是十分繁琐的,如果使用匿名对象,我不关注这个对象是谁,这个对象是多少,我只关心能不能引用类内的成员函数,因此就可以这样使用,相比起使用定义对象的方法来看,这样的方法的生命周期更短

相关文章
|
2天前
|
C++
【C++】日期类Date(详解)②
- `-=`通过复用`+=`实现,`Date operator-(int day)`则通过创建副本并调用`-=`。 - 前置`++`和后置`++`同样使用重载,类似地,前置`--`和后置`--`也复用了`+=`和`-=1`。 - 比较运算符重载如`&gt;`, `==`, `&lt;`, `&lt;=`, `!=`,通常只需实现两个,其他可通过复合逻辑得出。 - `Date`减`Date`返回天数,通过迭代较小日期直到与较大日期相等,记录步数和符号。 ``` 这是236个字符的摘要,符合240字符以内的要求,涵盖了日期类中运算符重载的主要实现。
|
2天前
|
存储 编译器 C++
【C++】类和对象④(再谈构造函数:初始化列表,隐式类型转换,缺省值
C++中的隐式类型转换在变量赋值和函数调用中常见,如`double`转`int`。取引用时,须用`const`以防修改临时变量,如`const int& b = a;`。类可以有隐式单参构造,使`A aa2 = 1;`合法,但`explicit`关键字可阻止这种转换。C++11起,成员变量可设默认值,如`int _b1 = 1;`。博客探讨构造函数、初始化列表及编译器优化,关注更多C++特性。
|
2天前
|
编译器 C++
【C++】类和对象④(类的默认成员函数:取地址及const取地址重载 )
本文探讨了C++中类的成员函数,特别是取地址及const取地址操作符重载,通常无需重载,但展示了如何自定义以适应特定需求。接着讨论了构造函数的重要性,尤其是使用初始化列表来高效地初始化类的成员,包括对象成员、引用和const成员。初始化列表确保在对象创建时正确赋值,并遵循特定的执行顺序。
|
2天前
|
C语言 C++
【C++】日期类Date(详解)③
该文介绍了C++中直接相减法计算两个日期之间差值的方法,包括确定max和min、按年计算天数、日期矫正及计算差值。同时,文章讲解了const成员函数,用于不修改类成员的函数,并给出了`GetMonthDay`和`CheckDate`的const版本。此外,讨论了流插入和流提取的重载,需在类外部定义以符合内置类型输入输出习惯,并介绍了友元机制,允许非成员函数访问类的私有成员。全文旨在深化对运算符重载、const成员和流操作的理解。
|
2天前
|
定位技术 C语言 C++
C++】日期类Date(详解)①
这篇教程讲解了如何使用C++实现一个日期类`Date`,涵盖操作符重载、拷贝构造、赋值运算符及友元函数。类包含年、月、日私有成员,提供合法性检查、获取某月天数、日期加减运算、比较运算符等功能。示例代码包括`GetMonthDay`、`CheckDate`、构造函数、拷贝构造函数、赋值运算符和相关运算符重载的实现。
|
2天前
|
编译器 C++
【C++】类和对象③(类的默认成员函数:赋值运算符重载)
在C++中,运算符重载允许为用户定义的类型扩展运算符功能,但不能创建新运算符如`operator@`。重载的运算符必须至少有一个类类型参数,且不能改变内置类型运算符的含义。`.*::sizeof?`不可重载。赋值运算符`=`通常作为成员函数重载,确保封装性,如`Date`类的`operator==`。赋值运算符应返回引用并检查自我赋值。当未显式重载时,编译器提供默认实现,但这可能不足以处理资源管理。拷贝构造和赋值运算符在对象复制中有不同用途,需根据类需求定制实现。正确实现它们对避免数据错误和内存问题至关重要。接下来将探讨更多操作符重载和默认成员函数。
|
2天前
|
存储 编译器 C++
【C++】类和对象③(类的默认成员函数:拷贝构造函数)
本文探讨了C++中拷贝构造函数和赋值运算符重载的重要性。拷贝构造函数用于创建与已有对象相同的新对象,尤其在类涉及资源管理时需谨慎处理,以防止浅拷贝导致的问题。默认拷贝构造函数进行字节级复制,可能导致资源重复释放。例子展示了未正确实现拷贝构造函数时可能导致的无限递归。此外,文章提到了拷贝构造函数的常见应用场景,如函数参数、返回值和对象初始化,并指出类对象在赋值或作为函数参数时会隐式调用拷贝构造。
|
2天前
|
存储 编译器 C语言
【C++】类和对象②(类的默认成员函数:构造函数 | 析构函数)
C++类的六大默认成员函数包括构造函数、析构函数、拷贝构造、赋值运算符、取地址重载及const取址。构造函数用于对象初始化,无返回值,名称与类名相同,可重载。若未定义,编译器提供默认无参构造。析构函数负责对象销毁,名字前加`~`,无参数无返回,自动调用以释放资源。一个类只有一个析构函数。两者确保对象生命周期中正确初始化和清理。
|
5天前
|
存储 编译器 C语言
【C++航海王:追寻罗杰的编程之路】类与对象你学会了吗?(上)
【C++航海王:追寻罗杰的编程之路】类与对象你学会了吗?(上)
10 2
|
5天前
|
C++
C++职工管理系统(类继承、文件、指针操作、中文乱码解决)
C++职工管理系统(类继承、文件、指针操作、中文乱码解决)
7 0
C++职工管理系统(类继承、文件、指针操作、中文乱码解决)