草庐IT

C++篇----构造函数和析构函数

青山与你 2024-04-16 原文

在很多时候,当写了初始化,动态开辟的,需要写销毁函数,写了销毁函数之后,但是却忘记了调用这些函数,忘记调用初始化函数还好,编译器会报错,但是如果是忘记调用销毁函数,那么编译器是不会报错,但是不能说这个程序就没错哦了,反而有很大的问题,存在内存泄漏的问题,如和解决这样问题?这也是本文重点。C++增加了类的6个默认成员函数,本文先分享构造函数和析构函数。
构造函数–主要完成初始化工作,
析构函数–主要完成清理工作

文章目录


一、前言

在很多时候,当写了初始化,动态开辟的,需要写销毁函数,写了销毁函数之后,但是却忘记了调用这些函数,忘记调用初始化函数还好,编译器会报错,但是如果是忘记调用销毁函数,那么编译器是不会报错,但是不能说这个程序就没错哦了,反而有很大的问题,存在内存泄漏的问题,如和解决这样问题?这也是本文重点。C++增加了类的6个默认成员函数,本文先分享构造函数和析构函数。

构造函数–主要完成初始化工作
析构函数–主要完成清理工作

二、构造函数

构造函数是特殊的成员函数,虽然它的名字叫构造函数,但是它的主要任务不是开空间创建对象,而是初始化对象
特性:
1、函数名和类名相同。 如:类名Stack,那么构造函数的函数,名也为Stack
2、无返回值(也不需要void)。
3、对象实例化编译器自动调用对应的构造函数。对象在定义之后就会调用它的默认构造函数
4、构造函数可以重载。(构造函数虽然没有返回值,但是可以有参数)
5、如果类中没有显式定义构造函数,则c++编译器会自动生成一个无参的默认构造函数,一旦用户显式定义,编译器不再默认生成。

以日期类带大家了解

#include<iostream>
using namespace std;

class Date
{
public:
	Date(int year=2022, int month=4, int day=24)
	{
		_year = year;
		_month = month;
		_day = day;
	}

	void Print()
	{
		cout << _year << '-' << _month << '-' << _day << endl;
	}

	int _year;
	int _month;
	int _day;
};

int main()
{
	Date d1;
	d1.Print();
	return 0;
}

在对象定义之后并没有显式调用构造函数,但是在打印这个对象的时候却是有内容的,因为在对象定义时编译器自动调用它的默认构造函数,(默认构造函数是构造函数参数列表全缺省)


构造函数是没有返回值的,且它的函数名与类名相同,它还能有参数也可以没有参数
无参和参数列表全给缺省值这两个函数在同一个类中只能出现其中之一

构造函数支持重载

对重载构造函数的调用

当构造函数重载,一个构造函数无参数,一个构造函数参数列表全给缺省值,对象定义之后自动调用它的默认构造函数,编译器不知道调用谁,此时会发生调用歧义,所以得出,构造函数无参数和参数列表全给缺省值这两个在一个类中只能出现其中之一,一个出现另一个就不能出现。

构造函数参数列表全为缺省参数或者无参时,在对象定以之后会自动调用,那么如果构造函数参数列表不权威缺省参数或者不为缺省参数时,应当如何?

当构造函数参数列表不全是缺省值时


此时会显示d1这个对象并不存在默认构造函数,在定义之后不能自动调用它的构造函数了
那么在定义对象时就不能简简单单的定义了 而是对象+参数列表,此时编译器才会自动调用构造函数
但是就算构造函数参数列表全为缺省参数或者无参,也不能像这样

此时编译器并不知道这个对象是声明还是定义

Data d1;//当构造函数参数列表全是缺省值时,编译器在对象实例化时才会默认自动调用构造函数,不然的话就需要对其传参
对象+参数列表,此时就会自动调用构造函数

构造函数:要么是无参、要么是全缺省、要么是没有显式写编译器会自动生成默认构造函数(不传参就可以调用的函数就是默认构造函数)

构造函数替代了Init(初始化),可以不用调用Init也对这个对象初始化了,这样也就解决了因没有调用初始化带来的错误,
构造函数是给对象初始化,在对象定义时自动调用

但是如果当类中没有定义构造函数时,应该如何处理?

编译器会自动生成构造函数,但是自动生成的构造函数,编译器并没用对类中的内置类型成员初始化
但是也有一些编译器会将其内置类型成员初始化为0,不过大多数是不会处理的

内置类型:(int/double/指针等等)
自定义类型:自己定义的/struct/class等等定义的类型

编译器自动生成构造函数也并不是什么都不做,当我们在类中,不显式写构造函数,编译器会默认生成构造函数,类中内置类型不做处理,自定义类型会调用它的默认构造函数

以日期类中字自定义一个栈对象来看

#include<iostream>
using namespace std;
class Stack
{
public:
	Stack(int capacity = 4)
	{
		_a = (int*)malloc(sizeof(int) * capacity);
		if (_a == nullptr)
		{
			perror("malloc fail\n");
			return;
		}

		_top = 0;
		_capacity = capacity;

		cout << "_top" << ' ' << "_capacity" << endl;
	}
	int _capacity;
	int* _a;
	int _top;
};


class Data
{
public:
	void Print()
	{
		cout << _year << '-' << _month << '-' << _day << endl;
	}

	int _year;
	int _month;
	int _day;

	Stack _st;
};

int main()
{
	Data d1;
	d1.Print();

	return 0;
}

可以发现,自动生成了默认构造函数,这里类的内置类型成员被处理为0了,其实大多数编译器是不会对类的内置类型成员进行处理

当类中没有构造函数时,编译器会默认自动生成构造函数 但是并不会对类的内置类型成员做处理(初始化)
当类中有自定义类型成员时,编译器会默认自动调用自定义类型它的构造函数,
同时有些编译器也会将类的内置类型成员初始化为0,但是这只是有些编译器,并不是全部

那么哪种类可以让编译器默认自动生成构造函数哪些不能? 一般情况下,类只要有内置类型成员,理论上不能让编译器默认自动生成构造函数
一般情况下,如果类成员全是自定义类型成员,可以考虑让编译器默认自动生成构造函数(两个栈实现队列)

所以在c++11标准发布,也算是为其做了一些补丁
在类中内置类型成员在声明的时候给缺省值,此时编译器默认自动生成构造函数时
这些内置类型成员不再是随机值,而是这些缺省值

#include<iostream>
using namespace std;
class Stack
{
public:
	Stack(int capacity = 4)
	{
		_a = (int*)malloc(sizeof(int) * capacity);
		if (_a == nullptr)
		{
			perror("malloc fail\n");
			return;
		}

		_top = 0;
		_capacity = capacity;

		cout << "_top" << ' ' << "_capacity" << endl;
	}
	int _capacity;
	int* _a;
	int _top;
};


class Data
{
public:
	
	void Print()
	{
		cout << _year << '-' << _month << '-' << _day << endl;
	}

	//可以在声明时给其缺省值
	int _year = 1;
	int _month = 1;
	int _day = 1;

	Stack _st;
};

int main()
{
	Data d1;
	d1.Print();

	return 0;
}

类中没有定义构造函数,编译器会自动生成默认构造函数,再自动调用生成的构造函数,然后此时对类的内置类型成员给缺省值,此时就相当于是对类的内置类型成员处理

  1. 类中没有显式定义构造函数,编译器会自动生成一个无参的默认构造函数
  2. 内置类型成员不做处理(a 有些编译器会处理 b.c++11标准对其缝补,声明可以给缺省值初始化)
  3. 自定义类型成员,会自动去调用它的构造函数

关于构造函数
总结:

  1. 一般情况下,构造函数都需要我们自己显式写
  2. 内置类型成员都有缺省值,且这些缺省值初始化符合自己要求,不用显式写构造函数
  3. 类中成员全是自定义类型,且这些类型都定义默认构造函数,不用显式写构造函数

三、析构函数

析构函数与构造函数功能相反,它是对资源的清理,相当于销毁函数,析构函数是在定义的对象生命周期结束时自动调用析构函数,完成对资源的清理,这样就算忘了调用销毁函数也不会造成内存泄漏了

析构函数特性:

  1. 析构函数名是在类名前加 ~
  2. 无参数,无返回值
  3. 一个类只能有一个析构函数(不能重载,析构函数无参数),
    若是未显式定义,系统会自动生成默认析构函数。
  4. 对象生命周期结束时,编译器会自动调用析构函数
    5.若是未显式定义,系统会自动生成默认析构函数(生成的默认析构函数和自动生成的构造函数类似(a.内置类型成员不做处理 b.自定义类型会调用它的析构函数))

以栈为例

#include<iostream>
using namespace std;
class Stack
{
public:
	Stack(int capacity = 4)
	{
		_a = (int*)malloc(sizeof(int) * capacity);
		if (_a == nullptr)
		{
			perror("malloc fail\n");
			return;
		}

		_top = 0;
		_capacity = capacity;

		cout << "_top=" <<_top<< ' ' << "_capacity="<<_capacity << endl;
	}

	~Stack()
	{
		free(_a);
		_a = nullptr;
		_top = 0;
		_capacity = 0;
		cout << "_top=" << _top << ' ' << "_capacity=" << _capacity << endl;
	}
	int _capacity;
	int* _a;
	int _top;
};

int main()
{
	Stack st1;

	return 0;
}


调试转到反汇编,可以发现,是在对象生命周期结束,自动调用析构函数
类中没有显式写析构函数,编译器会自动生成一个默认析构函数,其实和构造函数差不多

析构函数总结

一般情况下,如果对类成员写的是静态的,那么就不用写析构函数。有动态申请资源的,就需要显式写析构函数释放资源。

类中成员没有动态申请资源,不需要写析构函数
类中需要释放资源的成员都是自定义类型,不需要写析构函数

有关C++篇----构造函数和析构函数的更多相关文章

  1. ruby - 在没有 sass 引擎的情况下使用 sass 颜色函数 - 2

    我想在一个没有Sass引擎的类中使用Sass颜色函数。我已经在项目中使用了sassgem,所以我认为搭载会像以下一样简单:classRectangleincludeSass::Script::FunctionsdefcolorSass::Script::Color.new([0x82,0x39,0x06])enddefrender#hamlengineexecutedwithcontextofself#sothatwithintemlateicouldcall#%stop{offset:'0%',stop:{color:lighten(color)}}endend更新:参见上面的#re

  2. ruby-on-rails - 在 ruby​​ 中使用 gsub 函数替换单词 - 2

    我正在尝试用ruby​​中的gsub函数替换字符串中的某些单词,但有时效果很好,在某些情况下会出现此错误?这种格式有什么问题吗NoMethodError(undefinedmethod`gsub!'fornil:NilClass):模型.rbclassTest"replacethisID1",WAY=>"replacethisID2andID3",DELTA=>"replacethisID4"}end另一个模型.rbclassCheck 最佳答案 啊,我找到了!gsub!是一个非常奇怪的方法。首先,它替换了字符串,所以它实际上修改了

  3. ruby - 在 Ruby 中有条件地定义函数 - 2

    我有一些代码在几个不同的位置之一运行:作为具有调试输出的命令行工具,作为不接受任何输出的更大程序的一部分,以及在Rails环境中。有时我需要根据代码的位置对代码进行细微的更改,我意识到以下样式似乎可行:print"Testingnestedfunctionsdefined\n"CLI=trueifCLIdeftest_printprint"CommandLineVersion\n"endelsedeftest_printprint"ReleaseVersion\n"endendtest_print()这导致:TestingnestedfunctionsdefinedCommandLin

  4. ruby - 在 Ruby 中按名称传递函数 - 2

    如何在Ruby中按名称传递函数?(我使用Ruby才几个小时,所以我还在想办法。)nums=[1,2,3,4]#Thisworks,butismoreverbosethanI'dlikenums.eachdo|i|putsiend#InJS,Icouldjustdosomethinglike:#nums.forEach(console.log)#InF#,itwouldbesomethinglike:#List.iternums(printf"%A")#InRuby,IwishIcoulddosomethinglike:nums.eachputs在Ruby中能不能做到类似的简洁?我可以只

  5. C51单片机——实现用独立按键控制LED亮灭(调用函数篇) - 2

    说在前面这部分我本来是合为一篇来写的,因为目的是一样的,都是通过独立按键来控制LED闪灭本质上是起到开关的作用,即调用函数和中断函数。但是写一篇太累了,我还是决定分为两篇写,这篇是调用函数篇。在本篇中你主要看到这些东西!!!1.调用函数的方法(主要讲语法和格式)2.独立按键如何控制LED亮灭3.程序中的一些细节(软件消抖等)1.调用函数的方法思路还是比较清晰地,就是通过按下按键来控制LED闪灭,即每按下一次,LED取反一次。重要的是,把按键与LED联系在一起。我打算用K1来作为开关,看了一下开发板原理图,K1连接的是单片机的P31口,当按下K1时,P31是与GND相连的,也就是说,当我按下去时

  6. ruby-on-rails - 将字符串转换为 ruby​​-on-rails 中的函数 - 2

    我需要一个通过输入字符串进行计算的方法,像这样function="(a/b)*100"a=25b=50function.something>>50有什么方法吗? 最佳答案 您可以使用instance_eval:function="(a/b)*100"a=25.0b=50instance_evalfunction#=>50.0请注意,使用eval本质上是不安全的,尤其是当您使用外部输入时,因为它可能包含注入(inject)的恶意代码。另请注意,a设置为25.0而不是25,因为如果它是整数a/b将导致0(整数)。

  7. ruby - 在 ruby​​ 中使用 .try 函数和 .map 函数 - 2

    我需要从json记录中获取一些值并像下面这样提取curr_json_doc['title']['genre'].map{|s|s['name']}.join(',')但对于某些记录,curr_json_doc['title']['genre']可以为空。所以我想对map和join()使用try函数。我试过如下curr_json_doc['title']['genre'].try(:map,{|s|s['name']}).try(:join,(','))但是没用。 最佳答案 你没有正确传递block。block被传递给参数括号外的方法

  8. ruby - 是否可以从也在该模块中的类内部调用模块函数 - 2

    在这段Ruby代码中:ModuleMClassC当我尝试运行时出现“'M:Module'的未定义方法'helper'”错误c=M::C.new("world")c.work但直接从另一个类调用M::helper("world")工作正常。类不能调用在定义它们的同一模块中定义的模块函数吗?除了将类移出模块外,还有其他解决方法吗? 最佳答案 为了调用M::helper,你需要将它定义为defself.helper;结束为了进行比较,请查看以下修改后的代码段中的helper和helper2moduleMclassC

  9. ruby - 将运算符传递给函数? - 2

    也许这听起来很荒谬,但我想知道这对Ruby是否可行?基本上我有一个功能...defadda,bc=a+breturncend我希望能够将“+”或其他运算符(例如“-”)传递给函数,这样它就类似于...defsuma,b,operatorc=aoperatorbreturncend这可能吗? 最佳答案 两种可能性:以方法/算子名作为符号:defsuma,b,operatora.send(operator,b)endsum42,23,:+或者更通用的解决方案:采取一个block:defsuma,byielda,bendsum42,23,

  10. ruby - 我可以在 Ruby 1.9.x 中使用无参数函数吗? - 2

    所以我正在研究RubyKoans,而且我遇到了一个我认为是ruby1.9.x特有的问题。deftest_calling_global_methods_without_parenthesesresult=my_global_method2,3assert_equal5,resultend我明白了:james@tristan:~/code/ruby_projects/ruby_koans$rake(in/home/james/code/ruby_projects/ruby_koans)cdkoans/home/james/.rvm/rubies/ruby-1.9.2-p180/bin/ru

随机推荐