- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章C++抛出和接收异常的顺序由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
异常(exception)是C++语言引入的错误处理机制。它 采用了统一的方式对程序的运行时错误进行处理,具有标准化、安全和高效的特点。C++为了实现异常处理,引入了三个关键字:try、throw、catch。异常由throw抛出,格式为throw[expression],由catch捕捉。Try语句块是可能抛出异常的语句块,它通常和一个或多个catch语句块连续出现.
try语句块和catch语句块必须相互配合,以下三种情况都会导致编译错误:
(1)只有try语句块而没有catch语句块,或者只有catch语句块而没有try语句块; (2)在try语句块和catch语句块之间夹杂有其他语句; (3)当try语句块后跟有多个catch语句块时,catch语句块之间夹杂有其他语句; (4)同一种数据类型的传值catch分支与传引用catch分支不能同时出现.
在抛出和接收异常的过程中,我们还要注意以下几点.
1.被抛出的异常对象什么时候被销毁?
用throw语句抛出一个对象时,会构造一个新的对象,这个对象就是异常对象。该对象的生命周期从被抛出开始计算,一直到被某个catch语句捕捉,就会在该catch语句块执行完毕后被销毁。考察如下程序.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
|
#include <iostream>
using
namespace
std;
class
ExClass
{
int
num;
public
:
ExClass(
int
i)
{
cout<<
"Constructing exception object with num="
<<i<<endl;
num=i;
}
ExClass(ExClass& e)
{
cout<<
"Copy Constructing exception object with num="
<<e.num+1<<endl;
num=e.num+1;
}
~ExClass()
{
cout<<
"Destructing exception object with num="
<<num<<endl;
}
void
show()
{
cout<<
"the number is "
<<num<<endl;
}
};
int
main()
{
ExClass obj(99);
try
{
throw
obj;
//导致输出:Constructing exception object with num=100
}
catch
(
double
f)
{
cout<<
"exception catched"
<<endl;
}
//导致输出:Constructing exception object with num=101
catch
(ExClass e)
{
e.show();
}
cout<<
"after catch"
<<endl;
}
|
程序输出结果是:
Constructing exception object with num=99 Copy Constructing exception object with num=100 Copy Constructing exception object with num=101 the number is 101 Destructing exception object with num=101 Destructing exception object with num=100 after catch Destructing exception object with num=99 。
用throw语句抛出一个对象时,会构造一个新的对象,这个对象就是异常对象。该对象的生命周期从被抛出时开始计算,一直到被某个catch语句捕获,就会在该catch语句块执行完毕后被销毁。在上面的程序中,异常对象的num值为100,“Destructing exception object with num=100”这句话在“after catch”之前输出,正好说明异常对象的销毁时间是在它被捕获的catch块执行之后.
所以的catch分支在执行时类似一次函数调用,catch 的参数相当于函数的形参,而被抛出的异常对象相当于函数调用时的实参。当形参与实参成功匹配时,就说明异常被某个catch分支所捕获。catch后面的参数只能采用传值、传引用和传指针三种方式,如果采用传值方式,则会生成实参的一个副本,如果实参是一个对象,就会导致构造函数被调用。在上面的程序中,执行catch(ExClass e) 语句就是利用异常对象构造一个对象e,因此会调用拷贝构造函数。 要注意的是:同一种数据类型的传值catch分支和传引用catch分支不能同时出现.
2.异常如果在当前函数没有被捕获会发生什么?
在某些情况下,可能所有的catch分支都无法捕获到抛出的异常,这将导致当前函数执行的结束,并返回到主调函数中。在主调函数中,将继续以上的捕捉异常的过程,直到异常被捕捉或最终结束整个程序。考察如下程序.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
|
#include <iostream>
using
namespace
std;
class
ExClass
{
int
num;
public
:
ExClass(
int
i)
{
cout<<
"Constructing exception object with num="
<<i<<endl;
num=i;
}
ExClass(ExClass& e)
{
cout<<
"Copy Constructing exception object with num="
<<e.num+1<<endl;
num=e.num+1;
}
~ExClass()
{
cout<<
"Destructing exception object with num="
<<num<<endl;
}
void
show()
{
cout<<
"the number is "
<<num<<endl;
}
};
void
throwExFunc()
{
try
{
throw
ExClass(199);
}
catch
(
double
f){
cout<<
"double exception catched"
<<endl;
}
cout<<
"exit throwExFunc()"
<<endl;
}
int
main()
{
try
{
throwExFunc();
}
catch
(ExClass e)
{
e.show();
}
catch
(...)
{
cout<<
"all will fall in"
<<endl;
}
cout<<
"continue to execute"
<<endl;
}
|
程序的输出结果:
Constructing exception object with num=199 Copy Constructing exception object with num=200 the number is 200 Destructing exception object with num=200 Destructing exception object with num=199 continue to execute 。
从程序的结果可以看出:
(1)被抛出的异常对象的num值为199,由于它没有在函数throwExFunc()中被捕捉,所以它导致了throwExFunc()的执行结束(否则会输出:exit throwExFunc())。在main()函数中,catch(ExClass e)捕获了异常对象,通过复制构造函数产生对象e,e的num值为200,catch语句块运行完结束后,对象e首先被销毁,紧接着销毁异常对象。在这之后,程序继续运行,输出:continue to execute.
(2)catch(…)的意思是可以捕获所有类型的异常。不提倡随意地使用catch(…),因为这会导致异常类型的不精确处理,并降低程序的运行效率。但是,在程序的开发阶段,catch(…)还是有用的,因为如果在精心安排异常捕获之后,还是进入了catch(…)语句块,说明前面的代码存在缺陷,需要进一步改正.
(3)在捕捉异常对象时,还可以采用传引用的方式,例如把catch语句写成catch(ExClass& e),这样就可以不必产生异常对象的副本,减少程序的运行开销,提高运行效率.
(4)在抛出异常时,还可以抛出一个指针。当然这种做法并不总是安全的。如果要确保安全,应该将指针指向全局(静态)对象的指针或指向动态申请的空间,或者被抛出的指针在本函数内被捕获。否则,利用一个被抛出的指向已经被销毁的对象指针很危险。如果实在要用,首先,必须保证对象的析构函数不能对对象的内容作损伤性的修改,其次,对象的空间没有被其他新产生的变量覆盖。也就说,尽管对象被释放,但它的有效内容依然保留在栈中.
以上就是C++抛出和接收异常的顺序的详细内容,更多关于C++抛出和接收异常的资料请关注我其它相关文章! 。
原文链接:https://cloud.tencent.com/developer/article/1394311 。
最后此篇关于C++抛出和接收异常的顺序的文章就讲到这里了,如果你想了解更多关于C++抛出和接收异常的顺序的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
来自 java docs 公共(public) FileWriter(String fileName) 抛出 IOException 抛出: IOException - 如果指定的文件存在但它是目录而
我使用以下代码将我的 .net 客户端(基于 CQL)连接到 3 节点 Cassandra 集群。我以 30 条记录/秒的速度(从 RabbitMQ)获取数据,并且它们顺利地存储在 cassandra
如果在读取文件时缺少字段,我应该捕获 NoSuchElementException。如果缺少一个字段,我只需要跳到文件的下一行。我的问题是,我在哪里实现我的 try/catch 代码来做到这一点?这是
我正在尝试使用 ASP.NET MVC 实现 OpeinID 登录。我正在尝试按照 http://blog.nerdbank.net/2008/04/add-openid-login-support-
学习使用 Java 进行 xml 解析,并且正在编写一个测试程序来尝试各种东西。所有测试 System.out.println() 都是我在控制台中所期望的,除了 childElement 返回 [n
我正在尝试使用 SwingUtilities 创建 JFrame Thread tt = new Thread(new Runnable() { public void run
关闭。这个问题是not reproducible or was caused by typos .它目前不接受答案。 这个问题是由于错别字或无法再重现的问题引起的。虽然类似的问题可能是on-topi
我写了这段代码: MethodInfo method2 = typeof(IntPtr).GetMethod( "op_Explicit", Bind
我开始学习 Java,并且正在根据书本做一些练习。在执行此操作时,我遇到了以下错误:线程“main”java.util.InputMismatchException 中出现异常。我正在编写一个简单的程
我有一个文本文件,其中前两行是整数 m 和 n,然后有 m 行,每行都有 n 管道分隔值。我编写了一个程序,读取文件并使用文件中的值创建 m*n 数组,它工作了无数次,然后突然,使用相同的代码,使用相
所以我尝试使用在另一个类中生成的 bean 以在主应用程序中使用 package com.simon.spring.basics.properties; import org.spri
我还没有完成这个应用程序,但我希望在我的手机上看到它的样子。但是,它会强制关闭并引发 InstantiationException。 logcat 异常: 09-19 20:13:47.987: D/
我想从 UIViewController 加载一个基于 SwiftUI 的 View ,该 View 读取包本地的 json。仅 swiftUI 项目中的代码和绑定(bind)工作正常,当我利用 UI
'java.net.SocketTimeoutException:连接超时' 循环一段时间后我收到此错误。为什么我会收到 SocketTimeoutException?我该如何修复这个错误? @Ove
当有 null 值时抛出 ArgumentNullException() 是个好主意吗? This thread 没有提到在 null 上抛出的最明显的异常。 谢谢 最佳答案 ArgumentNull
我得到这个异常: NullReferenceException Object reference not set to an instance of an object at Namespace
所以其中一个方法的描述如下: public BasicLinkedList addToFront(T data) This operation is invalid for a sorted list
我正在使用 Intellij Idea,当我去生成 JavaDocs(通过工具 -> 生成 JavaDoc)时,我抛出了一个 IllegealArgumentException,没有关于发生了什么问题
我正在学习 C++ 中的互斥锁,但以下代码(摘自 N. Josuttis 的“C++ 标准库”)有问题。 我不明白为什么它会阻塞/抛出除非我在主线程中添加this_thread::sleep_for(
我正在试验 JavaFX 标签和组,通过鼠标拖动将它们移动到屏幕上。新节点从一些线程添加到动画组。但是,有时我会突然看到以下异常 - 我假设,当某些节点重叠时。但是不知道是什么问题……因为不涉及我的代
我是一名优秀的程序员,十分优秀!