C++异常处理

一 C++异常处理机制

  异常处理基本思想:执行一个函数的过程中发现异常,可以不用再本函数内立即进行处理,而是抛出该异常,让函数的调用者直接或间接的处理这个问题。

  C++异常处理进制由三个模块组成:try、throw、catch

  try---catch基本语法:

try 
 { 
可能抛出异常的语句;(检查) 
} 
catch(类型名[形参名])//捕获特定类型的异常 
{ 
 //处理1; 
} 
catch(类型名[形参名])//捕获特定类型的异常 
{ 
 //处理2; 
} 
catch(…)//捕获所有类型的异常 
{ 
 } 

   示例:

void f1(int i)
{
    if (i<0)
        throw 1;
}
void f2()
{
    f1(-45);
}
void f3()
{
    f2();
}
void f4()
{
    f3();
}
int main()
{
    try{
        f4();
    }
    catch (int i)
    {
        cout <<"error code:"<< i <<"\n"<< endl;
    }
    system("pause");
    return 0;
}

二 异常捕获的匹配规则

  try块里面抛出哪种异常,则catch里面捕获哪种类型的异常,一般情况下,类型必须完全匹配,但以下三种情况可以进行类型转换:

  (1)运行从非const对象到const对象转换,即throw一个非const对象,catch一个const对象

  (2)运行从派生类类型到基类型的转换

  (3)将数组转换为数组类型的指针,函数转换为指向函数类型的指针

#include <math.h>

bool Equal(double num1, double num2)
{
    if (num1 - num2 > -0.0000001 &&
        num1 - num2 < 0.0000001)
    {
        return true;
    }
    else
    {
        return false;
    }
}

// 基数为base为double类型,exponent为整数
double Power(double base, int exponent) // throw(char *)
{
    // 当底数为0且指数为负数时抛出异常
    if (Equal(base,0.0) && (exponent < 0))
    {
        throw exception("base must be positive!"); // thror "error"
    }
    double dbResult = 1.0;
    if (exponent >= 1)
    {
        for (int i = 0; i < exponent; i ++)
        {
            dbResult *= base;
        }
    }
    else if (exponent < 0)
    {
        for (int j = 0; j < abs(exponent); j ++)
        {
            dbResult *= base;
        }
        dbResult = 1/dbResult;
    }
    else if(exponent == 0)
    {
        dbResult = 1;
    }
    return dbResult;
}

void main()
{
    clock_t start,end;
    start = clock();
    try
    {
        cout << Power(0,1) << endl;
        cout << Power(2, 3) << endl;
        cout << Power(-2, 3) << endl;
        cout << Power(2, -3) << endl;
        cout << Power(2, 0) << endl;
        cout << Power(0,-1) << endl;
    }
    catch (exception & e) // char *p
    {
        cout << e.what() << endl;  // cout << p << endl;
    }
    end = clock();
    cout << "程序运行时间:" << (double)(end - start) / CLOCKS_PER_SEC << endl;
    return;
}

三 异常规范

  (1)函数声明之后,列出函数可能抛出的异常类型,并保证该函数不会抛出其他类型的异常

    a. void fun() throw(toobig,toosmall,divzero); // 函数抛出指定异常

    b. void f();  // 这意味函数会抛出任何一种异常

    c. void f() throw();   // 这意味函数不会抛出异常

  (2) 保证不在构造函数中抛出异常,因为构造函数的作用是构造对象并初始化,一旦抛出异常,有可能导致对象不完整或没有完全初始化

  (3)保证不在析构函数中抛出异常,因为析构函数的作用是完成资源的清理,一旦抛出异常,有可能导致内存泄露等等

四 使用异常的开销

  当异常被抛出时有相当的运行时间方面的开销,这就是从来不想把异常用于普通流控制的一部分的原因,而不管它多么令人心动。设计异常处理的重要目标之一是:在异常处理实现中,当异常不发生时应不影响运行速度,这就是说,只要不抛出异常,代码的允许速度如同没有加载异常处理一样,无论与否,异常处理都依赖于使用的特定编译器,异常处理会引出额外信息,这些信息被编译器置于栈上。

posted @ 2018-04-16 15:12  Fate0729  阅读(305)  评论(0编辑  收藏  举报