当前位置: 首页 > news >正文

阿里大鱼 wordpress/优化手机流畅度的软件

阿里大鱼 wordpress,优化手机流畅度的软件,烟台网站建设科技,资阳地seoC异常处理机制详解C自身有着非常强的纠错能力,发展到如今,已经建立了比较完善的异常处理机制. C的异常情况无非两种,一种是语法错误,即程序中出现了错误的语句、函数结构和类,致使编译程序无法进行。另一种是运行时发生…
f0dc813763ce26fb39a44dc8d99cc2e8.png

C++异常处理机制详解

C++自身有着非常强的纠错能力,发展到如今,已经建立了比较完善的异常处理机制. C++的异常情况无非两种,一种是语法错误,即程序中出现了错误的语句、函数结构和类,致使编译程序无法进行。另一种是运行时发生的错误,一般与算法有关。

关于语法错误, C++编译器的报错机制可以让我们轻松地解决这些错误; 第二种是运行时的错误,常见的有文件打开失败、数组下标溢出、系统内存不足等等。而一旦出现这些问题,引发算法失效、程序运行时无故停止等故障也是常有的。这就要求我们在设计软件算法时要全面。比如针对文件打开失败的情况,保护的方法有很多种,最简单的就是使用“return”命令,告诉上层调用者函数执行失败;另外一种处理策略就是利用c++的异常机制,抛出异常。

C++异常处理机制:

C++异常处理机制是一个用来有效地处理运行错误的非常强大且灵活的工具,它提供了更多的弹性、安全性和稳固性,克服了传统方法所带来的问题. 异常的抛出和处理主要使用了以下三个关键字: try、 throw 、 catch 。抛出异常即检测是否产生异常,在C++中,其采用throw语句来实现,如果检测到产生异常,则抛出异常。该语句的格式为:

throw 表达式;

如果在try语句块的程序段中(包括在其中调用的函数)发现了异常,且抛弃了该异常,则这个异常就可以被try语句块后的某个catch语句所捕获并处理,捕获和处理的条件是被抛弃的异常的类型与catch语句的异常类型相匹配。由于C++使用数据类型来区分不同的异常,因此在判断异常时,throw语句中的表达式的值就没有实际意义,而表达式的类型就特别重要。 try-catch语句形式如下:

try

{

包含可能抛出异常的语句;

}

catch(类型名 [形参名]) // 捕获特定类型的异常

{

//处理1

}

catch(类型名 [形参名]) // 捕获特定类型的异常

{

//处理2

}

catch(...) // 三个点则表示捕获所有类型的异常

{

}

【范例1】处理除数为0的异常。该范例将上述除数为0的异常可以用try/catch语句来捕获异常,并使用throw语句来抛出异常,从而实现异常处理,实现代码如代码如下:

#include //包含头文件

#include

double fuc(double x, double y) //定义函数

{

if(y==0)

{

throw y; //除数为0,抛出异常

}

return x/y; //否则返回两个数的商

}

void main()

{

double res;

try //定义异常

{

res=fuc(2,3);

cout<

res=fuc(4,0); 出现异常,函数内部会抛出异常

}

catch(double) //捕获并处理异常

{

cerr<

exit(1); //异常退出程序

}

}

【范例2】自定义异常类型

#include "stdafx.h"

#include

#include

#include

// 内存泄露检测机制

#define _CRTDBG_MAP_ALLOC

#ifdef _DEBUG

#define new new(_NORMAL_BLOCK, __FILE__, __LINE__)

#endif

// 自定义异常类

class MyExcepction

{

public:

// 构造函数,参数为错误代码

MyExcepction(int errorId)

{

// 输出构造函数被调用信息

std::cout << "MyExcepction is called" << std::endl;

m_errorId = errorId;

}

// 拷贝构造函数

MyExcepction( MyExcepction& myExp)

{

// 输出拷贝构造函数被调用信息

std::cout << "copy construct is called" << std::endl;

this->m_errorId = myExp.m_errorId;

}

~MyExcepction()

{

// 输出析构函数被调用信息

std::cout << "~MyExcepction is called" << std::endl;

}

// 获取错误码

int getErrorId()

{

return m_errorId;

}

private:

// 错误码

int m_errorId;

};

int main(int argc, char* argv[])

{

// 内存泄露检测机制

_CrtSetDbgFlag( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF );

// 可以改变错误码,以便抛出不同的异常进行测试

int throwErrorCode = 110;

std::cout << " input test code :" << std::endl;

std::cin >> throwErrorCode;

try

{

if ( throwErrorCode == 110)

{

MyExcepction myStru(110);

// 抛出对象的地址 -> 由catch( MyExcepction* pMyExcepction) 捕获

// 这里该对象的地址抛出给catch语句,不会调用对象的拷贝构造函数

// 传地址是提倡的做法,不会频繁地调用该对象的构造函数或拷贝构造函数

// catch语句执行结束后,myStru会被析构掉

throw &myStru;

}

else if ( throwErrorCode == 119 )

{

MyExcepction myStru(119);

// 抛出对象,这里会通过拷贝构造函数创建一个临时的对象传出给catch

// 由catch( MyExcepction myExcepction) 捕获

// 在catch语句中会再次调用通过拷贝构造函数创建临时对象复制这里传过去的对象

// throw结束后myStru会被析构掉

throw myStru;

}

else if ( throwErrorCode == 120 )

{

// 不提倡这样的抛出方法

// 这样做的话,如果catch( MyExcepction* pMyExcepction)中不执行delete操作则

//会发生内存泄露由catch( MyExcepction* pMyExcepction) 捕获

MyExcepction * pMyStru = new MyExcepction(120);

throw pMyStru;

}

else

{

// 直接创建新对象抛出

// 相当于创建了临时的对象传递给了catch语句

// 由catch接收时通过拷贝构造函数再次创建临时对象接收传递过去的对象

// throw结束后两次创建的临时对象会被析构掉

Throw MyExcepction(throwErrorCode);

}

}

catch( MyExcepction* pMyExcepction)

{

// 输出本语句被执行信息

std::cout << "执行了 catch( MyExcepction* pMyExcepction) " << std::endl;

// 输出错误信息

std::cout << "error Code : " << pMyExcepction->getErrorId()<< std::endl;

// 异常抛出的新对象并非创建在函数栈上,而是创建在专用的异常栈上,不需要进行delete

//delete pMyExcepction;

}

catch ( MyExcepction myExcepction)

{

// 输出本语句被执行信息

std::cout << "执行了 catch ( MyExcepction myExcepction) " << std::endl;

// 输出错误信息

std::cout << "error Code : " << myExcepction.getErrorId()<< std::endl;

}

catch(...)

{

// 输出本语句被执行信息

std::cout << "执行了 catch(...) " << std::endl;

// 处理不了,重新抛出给上级

throw ;

}

// 暂停

int temp;

std::cin >> temp;

return 0;

}

异常的接口声明

为了加强程序的可读性,使函数的用户能够方便地知道所使用的函数会抛出哪些异

常,可以在函数的声明中列出这个函数可能抛出的所有异常类型,例如:

void fun() throw( A,B,C,D);

这表明函数fun()可能并且只可能抛出类型(A,B,C,D)及其子类型的异常。

如果在函数的声明中没有包括异常的接口声明,则此函数可以抛出任何类型的异常,例如:

void fun();

一个不会抛出任何类型异常的函数可以进行如下形式的声明:

void fun() thow();

异常处理中需要注意的问题

(1). 如果抛出的异常一直没有函数捕获(catch),则会一直上传到c++运行系统那里,导致

整个程序的终止。

(2). 一般在异常抛出后资源可以正常被释放,但注意如果在类的构造函数中抛出异常,系

统是不会调用它的析构函数的,处理方法是:如果在构造函数中要抛出异常,则在抛出

前要记得删除申请的资源。

(3). 异常处理仅仅通过类型而不是通过值来匹配的,所以catch块的参数可以没有参数名

称,只需要参数类型。

(4). 函数原型中的异常说明要与实现中的异常说明一致,否则容易引起异常冲突。

(5). 应该在throw语句后写上异常对象时,throw先通过Copy构造函数构造一个新对

象,再把该新对象传递给 catch. 那么当异常抛出后新对象如何释放?

异常处理机制保证:异常抛出的新对象并非创建在函数栈上,而是创建在专用的异

常栈上,因此它才可以跨接多个函数而传递到上层,否则在栈清空的过程中就会被销

毁。所有从try到throw语句之间构造起来的对象的析构函数将被自动调用。但如果

一直上溯到main函数后还没有找到匹配的catch块,那么系统调用terminate()终止

整个程序,这种情况下不能保证所有局部对象会被正确地销毁。

(6). catch块的参数推荐采用地址传递而不是值传递,不仅可以提高效率,还可以利用对

象的多态性。另外,派生类的异常扑获要放到父类异常扑获的前面,否则,派生类的异

常无法被扑获。

(7). 编写异常说明时,要确保派生类成员函数的异常说明和基类成员函数的异常说明一

致,即派生类改写的虚函数的异常说明至少要和对应的基类虚函数的异常说明相同,甚

至更加严格,更特殊。

————————————————

a46ab1dd91c1edb95d0f2e8e0212dbc8.png

通过分享实用的计算机编程语言干货,推动中国编程到2025年基本实现普及化,使编程变得全民皆知,最终实现中国编程之崛起,这里是中国编程2025,感谢大家的支持。

原文链接:https://blog.csdn.net/dengzhouit/article/details/7012528

http://www.jmfq.cn/news/5054221.html

相关文章:

  • 如何做服装微商城网站建设/免费推广网址
  • 政务网站建设工作总结/揭阳新站seo方案
  • 网站改版的前端流程/sem代运营托管公司
  • 珠海外贸网站建设/合肥seo关键词排名
  • 做网站 哪些公司/站长工具查询网站信息
  • 局域网网站域名怎么做/百度地图人工客服电话
  • 做淘宝客网站 首选霍常亮/武汉seo和网络推广
  • 网站上传小马后怎么做/nba排名最新
  • pc网站建设怎么样/谷歌优化方法
  • 江苏省网站备案查询/站长工具seo下载
  • 漳州正规网站建设哪家便宜/你就知道
  • 设计网站 问题/seo优化技术
  • 网站内页banner一般做多高/指数基金
  • wordpress 判断文章页/谷歌seo教程
  • 枸橼酸西地那非片是什么/天津百度seo排名优化
  • 天津建设工程注册中心网站/实时新闻热点
  • 网站建设案例模板/杭州网站优化效果
  • 深圳网站科技有限公司是真是假/seo网站制作优化
  • 贵阳网站建设q479185700棒/拓客团队怎么联系
  • 北京信息网站建设/seo网站监测
  • 自己的电脑做服务器搭建网站/新闻热点大事件
  • 沈阳网页设计制作/搜索引擎排名优化方案
  • 做网站需要花费那方面的钱/seo网站推广专员
  • 安阳专业做网站公司/做销售有什么技巧和方法
  • 数据统计网站有哪些/做网站价格
  • 邢台网站建设地方/免费站长工具
  • 大连餐饮网站建设/网络媒体推广报价
  • asp网站建设mdb文件/发新闻稿平台
  • 网站降权查询/自己怎么做游戏推广赚钱
  • 做色情网站牟利200万判刑/百度风云排行榜