C++中的抛出异常
2022/1/14 22:07:58
本文主要是介绍C++中的抛出异常,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
以下为本人大一时阅读《C++ Primer Plus》中关于抛出异常章节所做的笔记
目录
梗概(下文会有详细内容):
try、catch的使用:
throw:
组成和继承的概念
类的作用域与类成员的访问:
访问函数(access fnction)和工具函数(utility function):
使用设置函数、获取函数的好处:
析构函数(destructor):
exit函数与abort函数:
左值:
const对象和const成员函数:
组成(composition):将对象作为类的成员
friend函数(友元函数)和friend类:
重载友元函数:
this指针:
static类成员:
与结构体类似,对象的名称或者对象的引用可以和原点成员选择符(.)一起使用,而对象的指针则需要和箭头成员选择运算符(->)一起使用
梗概(下文会有详细内容):
判定函数(predicate function):用于测试条件是真还是假
工具函数(也称为助手函数):类的private成员函数,目的是支持类的public成员函数的操作,并非为类的客户使用而准备的
析构函数(destructor):在类的对象撤销之前,用于完成对该对象的“扫尾工作”。
防止一个头文件在一个程序中被多次包含:
#ifndef TIME_H #define TIME_H ... #endif
ifndef:if no define(如果没有定义)
如果之前没有在文件中包含此头文件,那么TIME_H这个名字将被#define指令定义,并且包含该头文件的语句;
如果之前已包含此头文件,那么将不再包含该头文件
try、catch的使用:
try { ... } catch( 指定的错误类型 &e) { cerr<<"Exception: "<<e.what()<<endl; //(catch语句块内部也可以写其它的指令) }
注意:代码中指定类型的异常为其它头文件中的能指定错误的类型,例如在头文件<stdexcept>中的out_of_range、invalid_argument等,后面的 &e 是声明一个接收引用的异常形参,用来实现与捕捉到的异常对象的交互,e.what()是通过调用它的what成员函数打印异常的错误信息。catch语句块可以处理指定类型的异常。
当执行到以上步骤时,运行try中的语句,若存在异常,则运行catch中的语句(一般是用cerr输出关于错误的信息),然后再接着运行catch之后的程序;若不存在异常,直接跳过catch语句。一个try可以有多个catch块来处理不同的异常。
注意:在try语句块中声明的任何变量在catch语句块中都超出了它们的作用域,也就是说在catch语句块中都不可访问。
throw:
黏性设置与非黏性设置:
类定义体内的成员函数被隐式地声明为inline:
组成和继承的概念
类的作用域与类成员的访问:
访问函数(access fnction)和工具函数(utility function):
具有默认实参的构造函数:
注意:每个类最多只有一个默认构造函数(但可以有多个构造函数)
与一般函数一样,构造函数也可以有默认实参:
Time类:
public: explicit Time(int=0,int=0,int=0);
Time类的成员函数定义:
Time::Time(int hour,int minute,int second) { cin>>hour>>minute>>second; }
初始化3个Time对象:
Time t1; Time t2(2); Time t3(12,25,42);
在C++11中可以使用列表初始器调用构造函数:
Time t1; Time t2{2}; Time t3{12,25,42};
或者写成
Time t1; Time t2={2}; Time t3={12,25,42};
上述三种初始化都实现了:t1使用了3个默认实参,t2指定了一个实参(按形参的顺序传递,之传入一个实参时是传给了hour),t3指定了三个实参
C++11:重载的构造函数和委托构造函数:
使用设置函数、获取函数的好处:
类的成员函数可以直接访问类的private数据,但一般都调用设置函数(如setTime)和获取函数(如getTime)来修改和访问类的private数据,这样做的好处是当需要修改设置和获取数据的方式时,只需修改设置函数和获取函数的内容,而不需要对所有涉及到操作和获取private数据的内容都进行修改。
析构函数(destructor):
命名:在类名之前添加(~)
当对象撤销时,类的析构函数会隐式地调用。
析构函数本身并不释放对象占用的内存空间,它只是在系统收回对象的内存空间之前执行扫尾工作,这样内存可以重新用于保存新的对象。
如果没有显式地提供析构函数,编译器会隐式生成一个“空的”析构函数
exit函数与abort函数:
exit函数迫使程序立即结束,不执行自动对象的析构函数。当程序检测到输入中有错误,或者程序要处理的文件不能打开时,常常使用这个函数来终止程序。
abort函数的执行情况与exit函数类似,但是迫使程序立即终止,不允许调用任何对象的析构函数
总结:abort函数立即终止程序,exit函数终止程序前还进行一些清理工作
对象的存储类别对调用析构函数的顺序的影响:
全局作用域内对象:在文件内任何其他函数(包括main函数)开始执行之前调用构造函数、在main函数执行结束时调用析构函数
局部对象:当程序的执行每次进入或者离开自动对象的作用域时,自动对象的构造函数或者析构函数会被调用。(若程序的终止是由调用exit函数或者abort函数完成的,那么自动对象的析构函数将不被调用)
static局部对象:构造函数只被调用一次,即在程序第一次执行到该对象的定义处时,而相应的析构函数的调用发生在main函数结束或者程序调用exit函数时。全局或static对象的撤销顺序与它们建立的顺序正好相反。如果用abort函数的调用终止程序,那么static对象的析构函数将不被调用。
左值:
=是赋值运算符,它的作用是将一个表达式的值赋给一个左值。一个表达式或者是一个左值,或者是一个右值。所谓左值是指一个能用于赋值运算左边的表达式。左值必须能够被修改,不能是常量。这里是用变量作左值,指针和引用也可以作左值。
如果函数返回的是一个const引用,那么这个引用不能用作可修改的左值。
返回private数据成员的引用或指针:
对象的引用就是该对象名称的别名,可以在赋值语句的左边使用。
在类的定义中:
public: int &badSetHour(int); private: int hour;
在类的成员函数定义中:
int &Time::badSetHour(int hh) { hour=hh; return hour; } int Time::getHour() { return hour; }
在main函数中:
int main() { Time t; int &hourRef=t.badSetHour(20); cout<<t.getHour()<<endl; hourRef=30; cout<<t.getHour()<<endl; t.badSetHour(12)=74; cout<<t.getHour()<<endl; }
输出:
20
30
74
总结:通过引用hourRef来修改private中的数据,其中hourRef在声明的同时已用调用t.badSetHour(20)返回的引用来进行初始化。在这个过程中展示了hourRef如何破坏了类的封装性——main函数中的语句不应该有访问该类的private数据的权利。最后使用badSetHour函数调用本身作为左值,将74赋给该函数返回的引用。这个过程中,hour的值先是被赋值为传入badSetHour函数的参数12,然后被返回为引用,再被修改为74
默认的逐个成员赋值:
赋值运算符(=)可以将一个对象赋给另一个类型相同的对象,默认情况下是将=号右边对象的每个数据成员逐个赋值给左边对象同一个数据成员。对于每个类,编译器都提供了一个默认的复制构造函数,可以将原始对象的每个成员复制到新对象的相应成员中。对象可以作为函数的实参进行传递,也可以由函数返回。这种传递和返回默认情况下是以按值传递的方式执行的。
const对象和const成员函数:
修改const对象的任何企图在编译时就会被发现,而不是等到执行期才导致错误
以下调用是允许的:
1.对非const对象调用非const成员函数
2.对非const对象调用const成员函数
3.对const对象调用const成员函数
注意:对const对象调用非const成员函数是不想允许的
组成(composition):将对象作为类的成员
以下这句话差不多可以跟函数的声明顺序一样理解
创建顺序、撤销顺序:
由内而外创建、由外而内撤销(例如,Date成员对象作为Employee成员对象中的成员,则Date成员对象先创建,Employee成员对象后创建,同时Date成员对象在Employee对象撤销后再撤销)
friend函数(友元函数)和friend类:
类的friend函数在类的作用域之外定义,却具有访问类的非public(以及public)成员的权限。单独的函数、整个类或其他类的成员函数都可以被声明为另一个类的友元。
friend的声明:
使用friend函数修改类的private数据;
友元声明可以出现在类的任何地方。按照惯例,友元声明首先出现在类的定义中。
class Count { friend void setX(Count &,int); //*** public: Count():x(0){} private: int x; }; void setX(Count &c,int val) { c.x=val; } int main() { Count counter; setX(counter,1); }
上面的程序中使用了友元函数setX将类成员对象counter中的private成员x进行了修改。如果将标记了***行中的友元声明去掉,就会出现错误信息(一般类之外的函数不能对类成员对象中的private成员进行修改)
(实验课后的总结:友元函数的编写要注意编写程序的顺序,详细可以参考C++文件中实验课的2.1.6和2.1.7以及https://blog.csdn.net/jw903/article/details/38864769)
重载友元函数:
this指针:
每个对象都可以使用this指针来访问自己的地址
对象的 this指针不是指针本身的一部分,占用的内存大小不会反应在对对象进行sizeof运算得到的结果中。
this指针作为一个隐式的参数(被编译器)传递给对象的每个非static成员函数
使用this指针来避免名字冲突:
一个常用的this指针的explicit应用是用来避免类数据成员和成员函数参数之间的名字冲突。
在下面的例子中,设类Time有数据成员hour:
void Time::setHour(int hour) { this->hour=hour; }
上面的程序将传进setHour函数的hour参数赋值给数据成员hour
this指针的类型:
隐式和显式使用this指针来访问对象的数据成员:
void Test::print() const { cout<<x<<endl; // 1 cout<<this->x<<endl; // 2 cout<<(*this).x<<endl; // 3 }
上述输出的都是Test成员对象中的数据成员x的值,第一个隐式地使用this指针,仅仅指明该数据成员的名称,第二个和第三个使用不同的表示法通过this指针访问x。第三个请注意:*this必须用括号括起来,后面再跟随圆点成员选择运算符( . ),这样的原因是圆点运算符具有比*运算符更高的优先级(也即是*this.x的含义与(*this).x的含义不同)
使用this指针实现串联的函数调用:
串联的函数调用也就是多个函数在同一条语句中被调用
下面给个例子:
Time类的声明中:
public: Time &setTime(int,int,int); Time &setHour(int);
Time类成员函数定义:
Time &Time::setTime(int h,int m,int s) { ... return *this; } Time &Time::setHour(int h) { ... return *this; }
main函数中调用串联的成员函数:
int main() { Time t; t.setHour(18).setMinute(30).setSecond(22); }
串联的成员函数调用过程解读:
t.setHour(18).setMinute(30).setSecond(22);
先求t.setHour(18)的值,返回对对象t的引用,作为此函数调用的值。
接下来转化为:
t.setMinute(30).setSecond(22);
再接着转化为:
t.setSecond(22);
static类成员:
对于类的每个对象来说一般都各自拥有类所有数据成员的一份副本。但是static数据成员仅有变量的一份副本供类的所有对象共享。
这篇关于C++中的抛出异常的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2025-01-12深入理解 ECMAScript 2024 新特性:Map.groupBy() 分组操作
- 2025-01-11国产医疗级心电ECG采集处理模块
- 2025-01-10Rakuten 乐天积分系统从 Cassandra 到 TiDB 的选型与实战
- 2025-01-09CMS内容管理系统是什么?如何选择适合你的平台?
- 2025-01-08CCPM如何缩短项目周期并降低风险?
- 2025-01-08Omnivore 替代品 Readeck 安装与使用教程
- 2025-01-07Cursor 收费太贵?3分钟教你接入超低价 DeepSeek-V3,代码质量逼近 Claude 3.5
- 2025-01-06PingCAP 连续两年入选 Gartner 云数据库管理系统魔力象限“荣誉提及”
- 2025-01-05Easysearch 可搜索快照功能,看这篇就够了
- 2025-01-04BOT+EPC模式在基础设施项目中的应用与优势