参考:   https://blog.csdn.net/xueluowutong/article/details/81257654

 在c++中,可以直接抛出异常之后自己进行捕捉处理,如:(这样就可以在任何自己得到不想要的结果的时候进行中断,比如在进行数据库事务操作的时候,如果某一个语句返回SQL_ERROR则直接抛出异常,在catch块中进行事务回滚(回滚怎么理解?))。

  1.  
    #include <iostream>
  2.  
    #include <exception>
  3.  
    using namespace std;
  4.  
    int main () {
  5.  
    try
  6.  
    {
  7.  
    throw 1;
  8.  
    throw "error";
  9.  
    }
  10.  
    catch(char *str)
  11.  
    {
  12.  
    cout << str << endl;
  13.  
    }
  14.  
    catch(int i)
  15.  
    {
  16.  
    cout << i << endl;
  17.  
    }
  18.  
    }

也可以自己定义异常类来进行处理:

  1.  
    #include <iostream>
  2.  
    #include <exception>
  3.  
    using namespace std;
  4.  
     
  5.  
    //可以自己定义Exception
  6.  
    class myexception: public exception
  7.  
    {
  8.  
    virtual const char* what() const throw()
  9.  
    {
  10.  
    return "My exception happened";
  11.  
    }
  12.  
    }myex;
  13.  
     
  14.  
    int main () {
  15.  
    try
  16.  
    {
  17.  
    if(true) //如果,则抛出异常;
  18.  
    throw myex;
  19.  
    }
  20.  
    catch (exception& e)
  21.  
    {
  22.  
    cout << e.what() << endl;
  23.  
    }
  24.  
    return 0;
  25.  
    }

同时也可以使用标准异常类进行处理:

  1.  
    #include <iostream>
  2.  
    #include <exception>
  3.  
    using namespace std;
  4.  
     
  5.  
    int main () {
  6.  
    try
  7.  
    {
  8.  
    int* myarray= new int[100000];
  9.  
    }
  10.  
    catch (exception& e)
  11.  
    {
  12.  
    cout << "Standard exception: " << e.what() << endl;
  13.  
    }
  14.  
    return 0;
  15.  
    }

一、简单的例子

首先通过一个简单的例子来熟悉C++ 的 try/catch/throw(可根据单步调试来熟悉,try catch throw部分是如何运行的):

  1.  
    #include <stdlib.h>
  2.  
    #include "iostream"
  3.  
    using namespace std;
  4.  
     
  5.  
    double fuc(double x, double y) //定义函数
  6.  
    {
  7.  
    if(y==0)
  8.  
    {
  9.  
    throw y; //除数为0,抛出异常
  10.  
    }
  11.  
    return x/y; //否则返回两个数的商
  12.  
    }
  13.  
     
  14.  
    int _tmain(int argc, _TCHAR* argv[])
  15.  
    {
  16.  
    double res;
  17.  
    try //定义异常
  18.  
    {
  19.  
    res=fuc(2,3);
  20.  
    cout<<"The result of x/y is : "<<res<<endl;
  21.  
    res=fuc(4,0); //出现异常
  22.  
    }
  23.  
    catch(double) //捕获并处理异常
  24.  
    {
  25.  
    cerr<<"error of dividing zero.\n";
  26.  
    exit(1); //异常退出程序
  27.  
    }
  28.  
    return 0;
  29.  
    }

catch 的数据类型需要与throw出来的数据类型相匹配的。

二、catch(...)的作用

  catch(…)能够捕获多种数据类型的异常对象,所以它提供给程序员一种对异常对象更好的控制手段,使开发的软件系统有很好的可靠性。因此一个比较有经验的程序员通常会这样组织编写它的代码模块,如下:

  1.  
    void Func()
  2.  
    {
  3.  
      try
  4.  
      {
  5.  
        // 这里的程序代码完成真正复杂的计算工作,这些代码在执行过程中
  6.  
        // 有可能抛出DataType1、DataType2和DataType3类型的异常对象。
  7.  
      }
  8.  
      catch(DataType1& d1)
  9.  
      {
  10.  
      }
  11.  
      catch(DataType2& d2)
  12.  
      {
  13.  
      }
  14.  
      catch(DataType3& d3)
  15.  
      {
  16.  
      }
  17.  
      /*********************************************************
  18.  
      注意上面try block中可能抛出的DataType1、DataType2和DataType3三
  19.  
      种类型的异常对象在前面都已经有对应的catch block来处理。但为什么
  20.  
      还要在最后再定义一个catch(…) block呢?这就是为了有更好的安全性和
  21.  
      可靠性,避免上面的try block抛出了其它未考虑到的异常对象时导致的程
  22.  
      序出现意外崩溃的严重后果,而且这在用VC开发的系统上更特别有效,因
  23.  
      为catch(…)能捕获系统出现的异常,而系统异常往往令程序员头痛了,现
  24.  
      在系统一般都比较复杂,而且由很多人共同开发,一不小心就会导致一个
  25.  
      指针变量指向了其它非法区域,结果意外灾难不幸发生了。catch(…)为这种
  26.  
      潜在的隐患提供了一种有效的补救措施。
  27.  
      *********************************************************/
  28.  
     
  29.  
      catch(…)
  30.  
      {
  31.  
      }
  32.  
    }

三、异常中采用面向对象的处理

首先看下面的例子:

  1.  
    void OpenFile(string f)
  2.  
    {
  3.  
    try
  4.  
      {
  5.  
      // 打开文件的操作,可能抛出FileOpenException
  6.  
      }
  7.  
      catch(FileOpenException& fe)
  8.  
      {
  9.  
      // 处理这个异常,如果这个异常可以很好的得以恢复,那么处理完毕后函数
  10.  
      // 正常返回;否则必须重新抛出这个异常,以供上层的调用函数来能再次处
  11.  
       // 理这个异常对象
  12.  
      int result = ReOpenFile(f);
  13.  
      if (result == false) throw;
  14.  
      }
  15.  
    }
  16.  
     
  17.  
    void ReadFile(File f)
  18.  
    {
  19.  
      try
  20.  
      {
  21.  
      // 从文件中读数据,可能抛出FileReadException
  22.  
      }
  23.  
      catch(FileReadException& fe)
  24.  
      {
  25.  
      // 处理这个异常,如果这个异常可以很好的得以恢复,那么处理完毕后函数
  26.  
      // 正常返回;否则必须重新抛出这个异常,以供上层的调用函数来能再次处
  27.  
      // 理这个异常对象
  28.  
      int result = ReReadFile(f);
  29.  
      if (result == false) throw;
  30.  
      }
  31.  
    }
  32.  
     
  33.  
    void WriteFile(File f)
  34.  
    {
  35.  
      try
  36.  
      {
  37.  
      // 往文件中写数据,可能抛出FileWriteException
  38.  
      }
  39.  
      catch(FileWriteException& fe)
  40.  
      {
  41.  
      // 处理这个异常,如果这个异常可以很好的得以恢复,那么处理完毕后函数
  42.  
      // 正常返回;否则必须重新抛出这个异常,以供上层的调用函数来能再次处理这个异常对象
  43.  
        int result = ReWriteFile(f);
  44.  
      if (result == false) throw;
  45.  
      }
  46.  
    }
  47.  
     
  48.  
    void Func()
  49.  
    {
  50.  
      try
  51.  
      {
  52.  
      // 对文件进行操作,可能出现FileWriteException、FileWriteException
  53.  
      // 和FileWriteException异常
  54.  
      OpenFile(…);
  55.  
      ReadFile(…);
  56.  
      WriteFile(…);
  57.  
      }
  58.  
      // 注意:FileException是FileOpenException、FileReadException和FileWriteException
  59.  
      // 的基类,因此这里定义的catch(FileException& fe)能捕获所有与文件操作失败的异
  60.  
      // 常。
  61.  
      catch(FileException& fe)
  62.  
      {
  63.  
       ExceptionInfo* ef = fe.GetExceptionInfo();
  64.  
      cout << “操作文件时出现了不可恢复的错误,原因是:”<< fe << endl;
  65.  
      }
  66.  
    }

下面是更多面向对象和异常处理结合的例子:

  1.  
    #include <iostream.h>
  2.  
    class ExceptionClass
  3.  
    {
  4.  
    char* name;
  5.  
    public:
  6.  
    ExceptionClass(const char* name="default name")
  7.  
    {
  8.  
    cout<<"Construct "<<name<<endl;
  9.  
    this->name=name;
  10.  
    }
  11.  
    ~ExceptionClass()
  12.  
    {
  13.  
    cout<<"Destruct "<<name<<endl;
  14.  
    }
  15.  
    void mythrow()
  16.  
    {
  17.  
    throw ExceptionClass("my throw");
  18.  
    }
  19.  
    }
  20.  
     
  21.  
    void main()
  22.  
    {
  23.  
    ExceptionClass e("Test");
  24.  
    try
  25.  
    {
  26.  
    e.mythrow();
  27.  
    }
  28.  
    catch(...)
  29.  
    {
  30.  
    cout<<”*********”<<endl;
  31.  
    }
  32.  
    }

这是输出信息: Construct Test Construct my throw Destruct my throw **************** Destruct my throw   (这里是异常处理空间中对异常类的拷贝的析构) Destruct Test ======================================

不过一般来说我们可能更习惯于把会产生异常的语句和要throw的异常类分成不同的类来写,下面的代码可以是我们更愿意书写的:

class ExceptionClass
{
public:
ExceptionClass(const char* name="Exception Default Class")
{
cout<<"Exception Class Construct String"<<endl;
}
~ExceptionClass()
{
cout<<"Exception Class Destruct String"<<endl;
}
void ReportError()
{
cout<<"Exception Class:: This is Report Error Message"<<endl;
}
}; class ArguClass
{
char* name;
public:
ArguClass(char* name="default name")
{
cout<<"Construct String::"<<name<<endl;
this->name=name;
}
~ArguClass()
{
cout<<"Destruct String::"<<name<<endl;
}
void mythrow()
{
throw ExceptionClass("my throw");
}
}; _tmain()
{
ArguClass e("haha");
try
{
e.mythrow();
}
catch(int)
{
cout<<"If This is Message display screen, This is a Error!!"<<endl; //这行不会执行
}
catch(ExceptionClass pTest)
{
pTest.ReportError();
}
catch(...)
{
cout<<"***************"<<endl;
}
}

输出Message: Construct String::haha Exception Class Construct String Exception Class Destruct String Exception Class:: This is Report Error Message Exception Class Destruct String Destruct String::haha

四、构造和析构中的异常抛出 先看个程序,假如我在构造函数的地方抛出异常,这个类的析构会被调用吗?可如果不调用,那类里的东西岂不是不能被释放了?

#include <iostream.h>
#include <stdlib.h> class ExceptionClass1
{
char* s;
public:
ExceptionClass1()
{
cout<<"ExceptionClass1()"<<endl;
s=new char[4];
cout<<"throw a exception"<<endl;
throw 18;
}
~ExceptionClass1()
{
cout<<"~ExceptionClass1()"<<endl;
delete[] s;
}
}; void main()
{
try
{
ExceptionClass1 e;
}
catch(...)
{}
}

结果为: ExceptionClass1() throw a exception

在这两句输出之间,我们已经给S分配了内存,但内存没有被释放(因为它是在析构函数中释放的)。应该说这符合实际现象,因为对象没有完整构造。
为了避免这种情况,我想你也许会说:应避免对象通过本身的构造函数涉及到异常抛出。即:既不在构造函数中出现异常抛出,也不应在构造函数调用的一切东西中出现异常抛出。 但是在C++中可以在构造函数中抛出异常,经典的解决方案是使用STL的标准类auto_ptr。
那么,在析构函数中的情况呢?我们已经知道,异常抛出之后,就要调用本身的析构函数,如果这析构函数中还有异常抛出的话,则已存在的异常尚未被捕获,会导致异常捕捉不到。

五、标准C++异常类

标准异常都派生自一个公共的基类exception。基类包含必要的多态性函数提供异常描述,可以被重载。下面是exception类的原型:

class exception
{
public:
exception() throw();
exception(const exception& rhs) throw();
exception& operator=(const exception& rhs) throw();
virtual ~exception() throw();
virtual const char *what() const throw();
}; C++有很多的标准异常类:
namespace std
{
//exception派生
class logic_error; //逻辑错误,在程序运行前可以检测出来 //logic_error派生
class domain_error; //违反了前置条件
class invalid_argument; //指出函数的一个无效参数
class length_error; //指出有一个超过类型size_t的最大可表现值长度的对象的企图
class out_of_range; //参数越界
class bad_cast; //在运行时类型识别中有一个无效的dynamic_cast表达式
class bad_typeid; //报告在表达试typeid(*p)中有一个空指针p //exception派生
class runtime_error; //运行时错误,仅在程序运行中检测到 //runtime_error派生
class range_error; //违反后置条件
class overflow_error; //报告一个算术溢出
class bad_alloc; //存储分配错误
}

标准库异常类定义在以下四个头文件中

1、exception头文件:定义了最常见的标准异常类,其类名为exception。只通知异常的产生,但不会提供更多的信息

2、stdexcept头文件定义了以下几种常见异常类

函数                                                 功能或作用

exception                                            最常见的问题

runtime_error                                     运行时错误:仅在运行时才能检测到的问题

range_error                                        运行时错误:生成的结果超出了有意义的值域范围

overflow_error                                    运行时错误:计算上溢

underflow_error                                  运行时错误:计算下溢

logic_error                                           逻辑错误:可在运行前检测到的问题

domain_error                                       逻辑错误:参数的结果值不存在

invalid_argument                                 逻辑错误:不合适的参数

length_error                                        逻辑错误:试图生成一个超出该类型最大长度的对象

out_of_range                                       逻辑错误:使用一个超出有效范围的值

3、new头文件定义了bad_alloc异常类型,提供因无法分配内存而由new抛出的异常

4、type_info头文件定义了bad_cast异常类型(要使用type_info必须包含typeinfo头文件)

下面是使用异常类的例子:

首先,我定义了几个异常类,这些类也可以从标准异常类进行派生,如下

class BadInitializers
{
public:
BadInitializers() {}
};
class OutOfBounds
{
public:
OutOfBounds(int i) { cout<<"Size "<<i<<" is illegal!!!"<<endl; }
};
class SizeMismatch
{
public:
SizeMismatch() {}
};

然后要在程序中需要的地方使用throw来抛出异常类,两个抛出异常类的例子如下

template <class T>
Array1D<T>::Array1D(int sz)
{
if(sz<0)
{
//throw BadInitializers();
throw invalid_argument("Size has to be bigger than 0!!!"); }
size=sz;
element=new T[size];
}
template <class T>
T &Array1D<T>::operator[](int i) const
{
if(i<0||i>=size)
{
throw OutOfBounds(i);
}
return element[i];
}

然后在主程序中使用try...catch...来捕获异常,并进行相应的处理,如下

try
{
int i=0;
Array1D<int> a1(5);
a1[0]=1;
a1[1]=3;
a1[2]=5;
a1[3]=7;
a1[4]=8;
Array1D<int> a2(a1);
for(i=0;i<a2.Size();i++)
{
cout<<a2[i]<<" ";
}
cout<<endl;
Array1D<int> a3(5);
a3=a1+a2;
cout<<a3;
}
catch(BadInitializers)
{
cout<<"Error:BadInitializers!!!"<<endl;
}
catch(OutOfBounds &e)
{
cout<<"Error:OutOfBounds!!!"<<endl;
}
catch(SizeMismatch &e)
{
cout<<"Error:SizeMismatch!!!"<<endl;
}
catch(invalid_argument &e)
{
cout<<"Error:"<<e.what()<<endl;
}
catch(...)
{
cout<<"An unknown error!!!"<<endl;
}

六、try finally使用

__try {    file://保护块 } __finally {   file://结束处理程序 } 在上面的代码段中,操作系统和编译程序共同来确保结束处理程序中的__f i n a l l y代码块能够被执行,不管保护体(t r y块)是如何退出的。不论你在保护体中使用r e t u r n,还是g o t o,或者是longjump,结束处理程序(f i n a l l y块)都将被调用。
我们来看一个实列:(返回值:10, 没有Leak,性能消耗:小)

DWORD Func_SEHTerminateHandle()
{
DWORD dwReturnData = 0;
HANDLE hSem = NULL;
const char* lpSemName = "TermSem";
hSem = CreateSemaphore(NULL, 1, 1, lpSemName);
__try
{
WaitForSingleObject(hSem,INFINITE);
dwReturnData = 5;
}
__finally
{
ReleaseSemaphore(hSem,1,NULL);
CloseHandle(hSem);
}
dwReturnData += 5;
return dwReturnData;
}

这段代码应该只是做为一个基础函数,我们将在后面修改它,来看看结束处理程序的作用: ==================== 在代码加一句:(返回值:5, 没有Leak,性能消耗:中下)

DWORD Func_SEHTerminateHandle()
{
DWORD dwReturnData = 0;
HANDLE hSem = NULL;
const char* lpSemName = "TermSem";
hSem = CreateSemaphore(NULL, 1, 1, lpSemName);
__try
{
WaitForSingleObject(hSem,INFINITE);
dwReturnData = 5;
return dwReturnData;
}
__finally
{
ReleaseSemaphore(hSem,1,NULL);
CloseHandle(hSem);
}
dwReturnData += 5;
return dwReturnData;
}

  在try块的末尾增加了一个return语句。这个return语句告诉编译程序在这里要退出这个函数并返回dwTemp变量的内容,现在这个变量的值是5。但是,如果这个return语句被执行,该线程将不会释放信标,其他线程也就不能再获得对信标的控制。可以想象,这样的执行次序会产生很大的问题,那些等待信标的线程可能永远不会恢复执行。   通过使用结束处理程序,可以避免return语句的过早执行。当return语句试图退出try块时,编译程序要确保finally块中的代码首先被执行。要保证finally块中的代码在try块中的return语句退出之前执行。在程序中,将ReleaseSemaphore的调用放在结束处理程序块中,保证信标总会被释放。这样就不会造成一个线程一直占有信标,否则将意味着所有其他等待信标的线程永远不会被分配CPU时间。   在finally块中的代码执行之后,函数实际上就返回。任何出现在finally块之下的代码将不再执行,因为函数已在try块中返回。所以这个函数的返回值是5,而不是10。   读者可能要问编译程序是如何保证在try块可以退出之前执行finally块的。当编译程序检查源代码时,它看到在try块中有return语句。这样,编译程序就生成代码将返回值(本例中是5)保存在一个编译程序建立的临时变量中。编译程序然后再生成代码来执行f i n a l l y块中包含的指令,这称为局部展开。更特殊的情况是,由于try块中存在过早退出的代码,从而产生局部展开,导致系统执行finally块中的内容。在finally块中的指令执行之后,编译程序临时变量的值被取出并从函数中返回。   可以看到,要完成这些事情,编译程序必须生成附加的代码,系统要执行额外的工作。

finally块的总结性说明 我们已经明确区分了强制执行finally块的两种情况: • 从try块进入finally块的正常控制流。 • 局部展开:从try块的过早退出(goto、long jump、continue、break、return等)强制控制转移到finally块。 第三种情况,全局展开( global unwind),这个以后再看。

七、C++异常参数传递

从语法上看,在函数里声明参数与在catch子句中声明参数是一样的,catch里的参数可以是值类型,引用类型,指针类型。例如:

try
{
.....
}
catch(A a)
{
}
catch(B& b)
{
}
catch(C* c)
{
}

  尽管表面是它们是一样的,但是编译器对二者的处理却又很大的不同。调用函数时,程序的控制权最终还会返回到函数的调用处,但是抛出一个异常时,控制权永远不会回到抛出异常的地方。

class A;
void func_throw()
{
A a;
throw a; //抛出的是a的拷贝,拷贝到一个临时对象里
}
try
{
func_throw();
}
catch(A a) //临时对象的拷贝
{
}

  当我们抛出一个异常对象时,抛出的是这个异常对象的拷贝。当异常对象被拷贝时,拷贝操作是由对象的拷贝构造函数完成的。该拷贝构造函数是对象的静态类型(static type)所对应类的拷贝构造函数,而不是对象的动态类型(dynamic type)对应类的拷贝构造函数。此时对象会丢失RTTI信息。   异常是其它对象的拷贝,这个事实影响到你如何在catch块中再抛出一个异常。比如下面这两个catch块,乍一看好像一样:

catch (A& w) // 捕获异常
{
 // 处理异常
 throw; // 重新抛出异常,让它继续传递
}
catch (A& w) // 捕获Widget异常
{
 // 处理异常
 throw w; // 传递被捕获异常的拷贝
}

  第一个块中重新抛出的是当前异常(current exception),无论它是什么类型。(有可能是A的派生类)     第二个catch块重新抛出的是新异常,失去了原来的类型信息。   一般来说,你应该用throw来重新抛出当前的异常,因为这样不会改变被传递出去的异常类型,而且更有效率,因为不用生成一个新拷贝。 看看以下这三种声明: catch (A w) ... // 通过传值 catch (A& w) ... // 通过传递引用,一个被异常抛出的对象(总是一个临时对象)可以通过普通的引用捕获 catch (const A& w) ... //const引用

  回到异常对象拷贝上来。我们知道,当用传值的方式传递函数的参数,我们制造了被传递对象的一个拷贝,并把这个拷贝存储到函数的参数里。同样我们通过传值的方式传递一个异常时,也是这么做的当我们这样声明一个catch子句时: catch (A w) ... // 通过传值捕获 会建立两个被抛出对象的拷贝,一个是所有异常都必须建立的临时对象,第二个是把临时对象拷贝进w中。实际上,编译器会优化掉一个拷贝。同样,当我们通过引用捕获异常时, catch (A& w) ... // 通过引用捕获 catch (const A& w) ... //const引用捕获 这仍旧会建立一个被抛出对象的拷贝:拷贝是一个临时对象。相反当我们通过引用传递函数参数时,没有进行对象拷贝。话虽如此,但是不是所有编译器都如此。

  另外,通过指针抛出异常与通过指针传递参数是相同的。不论哪种方法都是一个指针的拷贝被传递。你不能认为抛出的指针是一个指向局部对象的指针,因为当异常离开局部变量的生存空间时,该局部变量已经被释放。Catch子句将获得一个指向已经不存在的对象的指针。这种行为在设计时应该予以避免。   另外一个重要的差异是在函数调用者或抛出异常者与被调用者或异常捕获者之间的类型匹配的过程不同。在函数传递参数时,如果参数不匹配,那么编译器会尝试一个类型转换,如果存在的话。而对于异常处理的话,则完全不是这样。见一下的例子:

void func_throw()
{
CString a;
throw a; //抛出的是a的拷贝,拷贝到一个临时对象里
} try
{
func_throw();
}
catch(const char* s)
{
}

  抛出的是CString,如果用const char*来捕获的话,是捕获不到这个异常的。   尽管如此,在catch子句中进行异常匹配时可以进行两种类型转换。第一种是基类与派生类的转换,一个用来捕获基类的catch子句也可以处理派生类类型的异常。反过来,用来捕获派生类的无法捕获基类的异常。   第二种是允许从一个类型化指针(typed pointer)转变成无类型指针(untyped pointer),所以带有const void* 指针的catch子句能捕获任何类型的指针类型异常: catch (const void*) ... //可以捕获所有指针异常   另外,你还可以用catch(...)来捕获所有异常,注意是三个点。   传递参数和传递异常间最后一点差别是catch子句匹配顺序总是取决于它们在程序中出现的顺序。因此一个派生类异常可能被处理其基类异常的catch子句捕获,这叫异常截获,一般的编译器会有警告。

class A {
public:
A()
{
cout << "class A creates" << endl;
}
void print()
{
cout << "A" << endl;
}
~A()
{
cout << "class A destruct" << endl;
}
};
class B: public A
{
public:
B()
{
cout << "class B create" << endl;
}
void print()
{
cout << "B" << endl;
}
~B()
{
cout << "class B destruct" << endl;
}
};
void func()
{
B b;
throw b;
}
try
{
func();
}
catch( B& b) //必须将B放前面,如果把A放前面,B放后面,那么B类型的异常会先被截获。
{
b.print();
}
catch (A& a)
{
a.print() ;
}

C++中try&catch的更多相关文章

  1. 【转】Java中try catch finally语句中含有return语句的执行情况(总结版)

    Java中try catch finally语句中含有return语句的执行情况(总结版) 有一点可以肯定,finally块中的内容会先于try中的return语句执行,如果finall语句块中也有r ...

  2. c++中try catch的用法

    c++中try catch的用法 标签: c++exception数据库sqlc 2011-10-24 21:49 45622人阅读 评论(3) 收藏 举报  分类: 一点小结(267)  版权声明: ...

  3. PHP中try{}catch{}的具体用法详解

    PHP中try{}catch{}是异常处理,将要执行的代码放入TRY块中,如果这些代码执行过程中某一条语句发生异常,则程序直接跳转到CATCH块中,由$e收集错误信息和显示.任何调用 可能抛出异常的方 ...

  4. javascript中 try catch用法

    javascript中 try catch用法 投稿:hebedich 字体:[增加 减小] 类型:转载 时间:2015-08-16我要评论 JS try catch语句一般在什么情况下使用?是必须使 ...

  5. 在zend framework框架中try{}catch(Exception e){}的跳转问题

    请勿盗版,转载请加上出处http://blog.csdn.net/yanlintao1 首先我先说明我遇到的问题 try{ //导入学生信息 $ModelStudent->insert($dat ...

  6. Java中try,catch,finally的用法

    Java中try,catch,finally的用法,以前感觉还算熟悉,但看到一篇博文才有更深点的理解,总结网友博客如下. Java异常处理的组合方式: 1.try+catch  运行流程:运行到try ...

  7. C#中try catch中throw ex和throw方式抛出异常有何不同_异常捕获堆栈丢失问题

    前言,最近遇到一个使用try-catch异常捕获后记录一下日志,然后再抛出该异常后,异常堆栈里无法显示准确的堆栈地址的问题?   其实以前也遇到过类似问题,没有重视,这次好好研究了下,并上度娘上找了找 ...

  8. C++中Try Catch中的继承

    1.C++中Try Catch简介:我们编译运行程序出错的时候,编译器就会抛出异常.抛出异常要比终止程序灵活许多. 而C++异常是指在程序运行时发生的反常行为,这些行为超出了函数正常功能的范围.当程序 ...

  9. C#中try catch中throw ex和throw方式抛出异常有何不同

    我们在C#的try catch代码块中里面经常使用throw语句抛出捕捉到的异常,但是你知道吗使用throw ex和throw抛出捕获到的异常效果是不一样的. 异常捕捉的原理 首先先介绍一下C#异常捕 ...

  10. JS中try....catch

    1.事情还有得挽回,换条路走try { 执行某个逻辑} catch (e) { 出问题,换个逻辑执行} 2.体面的退出try { 正常流程} catch (e) { 弹个框告诉用户不好意思出了点问题 ...

随机推荐

  1. HDU - 1005 -Number Sequence(矩阵快速幂系数变式)

    A number sequence is defined as follows:  f(1) = 1, f(2) = 1, f(n) = (A * f(n - 1) + B * f(n - 2)) m ...

  2. Postman发送POST请求到Spring Boot的正确姿势

    最近用Spring Boot搭建了一些restful api,写起来真的很爽.但是当用Postman测试一些POST请求的接口的时候却遇到一些问题,上网冲浪查了一堆博客资料,发现都讲得不清不楚,于是记 ...

  3. Django模型验证器详解和源码分析

    转发请注明来源 在Django的模型字段参数中,有一个参数叫做validators,这个参数是用来指定当前字段需要使用的验证器,也就是对字段数据的合法性进行验证,比如大小.类型等. Django的验证 ...

  4. iptables实用知识 ,一文学会配置linux防火墙

    目录 1.防火墙的概念 2. linux防火墙 3.linux数据包处理流程 3.1 linux 防火墙将以上流程,固定区分为5个流程节点 3.2 数据流程 4 linux防火墙的实现机制 4.1 i ...

  5. 为什么ping不通google.com

    前言 为什么在ping不通Google的时候,我们却可以web直接访问Google (已开启SSR 翻 墙) SSR访问Google 因为GFW的限制导致国内无法直接访问谷歌,那么SSR为什么能绕过限 ...

  6. .NET委托,事件和Lambda表达式

    委托 委托是什么? 委托是一种引用类型(其实就是一个类,继承MulticastDelegate特殊的类.),表示对具有特定参数列表和返回类型的方法的引用. 每个委托提供Invoke方法, BeginI ...

  7. Java 8 Stream API实例

    一.开篇 Stream?其实就是处理集合的一种形式,称之为流,在Java8中被引入,可被Collection中的子类调用. 作用?简化代码,提升你的开发效率. 不会?看完这篇你就能自己上手了! 二.实 ...

  8. Python全栈工程师系列学习之学习记录

    @ 目录 前言 Day 01 一.python的历史和种类 二.安装python解释器以及配置环境变量 三.变量.常量和注释 Day 02 Day 03 Day 04 Day 05 Day 06 一. ...

  9. C#开发PACS医学影像处理系统(十三):绘图处理之病灶测量

    接上一篇文章,当我们可以绘制图形标记后,就可以在此操作类上面进行扩展, 比如测量类工具,目前整理出的常用绘图和测量功能如下: 测量工具类:(图形标记类请参考本系列文章:绘图处理之图形标记) 功能 说明 ...

  10. mariadb 2

    mariadb第二章-增删改   MariaDB 数据类型 MariaDB数据类型可以分为数字,日期和时间以及字符串值. 使用数据类型的原则:够用就行, 尽量使用范围小的,而不用大的 常用的数据类型 ...