Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 580|回复: 0

Visual Studio中使用异常方法及开启

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-7-13 08:55:36 | 显示全部楼层 |阅读模式

      在使用Windows程序的时候,相信没有用户喜欢看到程序突然崩溃的情形吧!为了避免程序崩溃,程序员在编写程序的时候最好在容易出错的地方使用异常处理机制来保证友好的用户体验。特别是编写C/C++代码的时候,崩溃是经常的事情!
      今天一位同事给我说,编写C/C++代码崩溃的原因主要是因为内存操作违规。如果每次操作一块内存或指针之前都检查内存或指针是否有效,那么可以降低程序崩溃的次数。但是这会让程序员很厌烦的,哈哈。所以在适当的地方加上异常处理,即使崩溃也会让程序员更好的改善程序。当然,程序效率必然降低! 

      幸好C++规范中有异常处理机制: try catch

      但是在Visual Studio中直接使用try catch是不能产生异常的,必须手动抛出异常throw,见如下代码:

    void TryCatchfourth()
    {
    char* Test = NULL;
    try
    {
    Test = new char[2];
    FreeArray(Test);
    throw 0; // 这里我是随便抛出异常以测试
    // 若没有手动throw,后面即使产生异常catch里面的代码还是不会被执行
    puts("No "); // 接下来这两句不会被执行
    *(Test + 4096 ) = '\0';
    }
    catch (...)
    {
    if (NULL != Test)
    {
    FreeArray(Test);
    }
    printf("4 \n");
    }
    puts("Fourth");
    }

      又下面这段代码,如果将vs编译器的选项修改:打开项目属性→配置属性→C/C++→代码生成→启用C++异常→是,但有SEH异常(/EHa) (这里编译器默认为"是(/EHsc)")。

    void TryCatchThree()
    {
    char* Test = NULL;
    try
    {
    Test = new char[2];
    FreeArray(Test);
    *(Test + 4096 ) = '\0';
    }
    catch (...)
    {
    if (NULL != Test)
    {
    FreeArray(Test);
    }
    printf("4 \n");
    }
    puts("Three");
    }

      修改编译器选项之后执行TryCatchThree()就正常工作。请注意TryCatchThree()和TryCatchfourth()所采用的不同编译选项。

      最后参考一个Windows的结构化异常处理代码:

    void TryCatchSecond()
    {
    char* Test = NULL;
    __try
    {
    __try
    {
    Test = new char[2];
    FreeArray(Test);
    *(Test + 4096 ) = '\0';
    // 此处数字一定要大,否则不能产生异常。
    // 因为Windows平台下,new N个字节,系统分配的一定比N大。
    // 4096大小刚好是一个页面大小,如果N>4096,那就自己扩大对应数字以产生异常
    }
    __finally
    {
    if (NULL != Test)
    {
    FreeArray(Test);
    }
    printf("2 "); // this is printed second
    }
    }
    __except ( FilterFunction(GetExceptionCode()) )
    {
    printf("3 \n"); // this is printed last
    }
    puts("Second");
    }

      不管编译器选项是否按照上述要求被修改,TryCatchSecond()均能正常工作。同事说,这是Windows结构化异常,和C++中的异常有点不一样,我现在也不太懂,以后查到资料后在添加到博客中来。TryCatchSecond()代码是参考MSDN中的一个例子,如下:

    DWORD FilterFunction(int i = 1) 
    {
    printf("%d ", i); // printed first
    return EXCEPTION_EXECUTE_HANDLER;
    }

    void TryCatchFirst()
    {
    char* Test = NULL;
    __try
    {
    __try
    {

    // 这个API是手动设置异常代码(这么称呼有点别扭)
    RaiseException(1, // exception code
    0, // continuable exception
    0, NULL); // no arguments
    }
    __finally
    {
    printf("2 "); // this is printed second
    }
    }
    __except ( FilterFunction() )
    {
    printf("3 \n"); // this is printed last
    }
    puts("One");
    }

      下面是完整代码:(本文中代码均在vs2008中编写和测试,建议不要在vc6下测试,vc6对C++规范支持很不好

    View Code
    #include <windows.h>

    // 释放数组内存
    #define FreeArray(pArray) { \
    delete[] pArray; \
    pArray = NULL; \
    }
    DWORD FilterFunction(int i = 1)
    {
    printf("%d ", i); // printed first
    return EXCEPTION_EXECUTE_HANDLER;
    }

    void TryCatchFirst()
    {
    char* Test = NULL;
    __try
    {
    __try
    {

    // 这个API是手动设置异常代码(这么称呼有点别扭)
    RaiseException(1, // exception code
    0, // continuable exception
    0, NULL); // no arguments
    }
    __finally
    {
    printf("2 "); // this is printed second
    }
    }
    __except ( FilterFunction() )
    {
    printf("3 \n"); // this is printed last
    }
    puts("One");
    }

    void TryCatchSecond()
    {
    char* Test = NULL;
    __try
    {
    __try
    {
    Test = new char[2];
    FreeArray(Test);
    *(Test + 4096 ) = '\0';
    // 此处数字一定要大,否则不能产生异常。
    // 因为Windows平台下,new N个字节,系统分配的一定比N大。
    // 4096大小刚好是一个页面大小,如果N>4096,那就自己扩大对应数字以产生异常
    }
    __finally
    {
    if (NULL != Test)
    {
    FreeArray(Test);
    }
    printf("2 "); // this is printed second
    }
    }
    __except ( FilterFunction(GetExceptionCode()) )
    {
    printf("3 \n"); // this is printed last
    }
    puts("Second");
    }
    void TryCatchThree()
    {
    char* Test = NULL;
    try
    {
    Test = new char[2];
    FreeArray(Test);
    *(Test + 4096 ) = '\0';
    }
    catch (...)
    {
    if (NULL != Test)
    {
    FreeArray(Test);
    }
    printf("4 \n");
    }
    puts("Three");
    }
    void TryCatchfourth()
    {
    char* Test = NULL;
    try
    {
    Test = new char[2];
    FreeArray(Test);
    throw 0; // 这里我是随便抛出异常以测试
    // 若没有手动throw,后面即使产生异常catch里面的代码还是不会被执行
    puts("No "); // 接下来这两句不会被执行
    *(Test + 4096 ) = '\0';
    }
    catch (...)
    {
    if (NULL != Test)
    {
    FreeArray(Test);
    }
    printf("4 \n");
    }
    puts("Fourth");
    }

    VOID main(VOID)
    {
    // TryCatchFirst();
    // TryCatchSecond();
    // TryCatchThree();
    // TryCatchfourth();
    }

     

    【参考资料 感谢作者】
    1、我的同事 



       




     

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-12-22 15:45 , Processed in 0.059447 second(s), 30 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表