一
前言
在类方法中处理业务逻辑时,对于异常情况需要中断类定义或者类方法的执行.同时这些异常情况还需要被调用点捕获,以便报错或者进入例外处理逻辑.
如果没有捕获这些错误,则会产生SHORT DUMP.
本文主要讲解类中触发异常的方式及调用类捕获异常的方式
二
异常处理方式
在ABAP类中异常处理有两种方式
这两种方式不能混用. 一个特定的类方法中只能使用一种方式.但是同一个类中的不同的方法可以各自使用一种方式
三
传统方式
传统方式与函数的异常处理类似.
01
触发异常
在类方法中添加异常, 通过RAISE 语句或者 MESSAGE RAISING 触发异常.


02
捕获异常
传统异常的捕获只能通过 调用方法时,传递EXCEPTION 参数. 特定异常返回一个值到 SY-SUBRC 中. 在调用点对 SY-SUBRC 内容进行判断,执行后续逻辑.


四
异常类方式
使用一个特殊的类-异常类来集中管理异常
01
创建异常类
创建一个异常类ZCX_TEST_EXCEPTION,创建的时候,系统识别ZCX_前缀, 会自动填充超类 CX_STATIC_CHECK ,且类型设置为异常类.异常类会自动生成类的构造方法代码且不可修改.

02
可能犯的错误
如果异常类的实例生成没有设置为2. 使用时会报下图的错误, 设置成公用后, 就不会报错了.

如果在类方法的异常中没有引入异常类, 语法检查报警告,捕获错误没有明确的信息

测试调用对象,捕获如下错误

03
引入异常类
需要在例外中引用该异常类

04
触发异常类
触发异常时可以传递参数, 也可以不传递参数,使用异常类中默认的消息

05
异常类的捕获


06
触发异常类传递的参数
MSGID MSGNO 用于识别一条消息. ATTR1-ATTR4 传递的并不是消息的变量内容. 而是类的属性 . 比如 IF_T100_DYN_MSG~MSGV1 或者自定义的属性
如果在ATTR1-ATTR4中传递一个字符串. 最终捕获的错误文本中 将出现 &字符串&
异常类中需要定义属性默认值. 这样才能通过参数传递所需的属性的名称. 最终使用消息+属性内容形成消息内容


五
通用异常类
如果希望系统生成的异常类传递错误消息内容不带&符号.可以通过添加异常类属性和改写 IF_MESSAGE~GET_TEXT 方法. 在RAISE EXCEPTION时传递一个消息描述
01
创建异常类
创建一个通用异常类 ZCX_BC_COMM

02
添加属性
添加属性 ERROR_TEXT
添加的属性会自动出现在构造方法中的参数中,并且会自动添加赋值到属性的语句

03
重定义方法
重定义方法 IF_MESSAGE~GET_TEXT

重定义方法的代码内容 优先获取 me->error_text 属性中的内容返回

04
触发通用异常类
触发异常类时,传递自定义的属性生成的参数

05
捕获的消息


六
总结
类异常的触发及捕获建议使用异常类的方式. 这样可以更加灵活的处理类异常情况.
捕获异常时,可以使用特定的异常类捕获. 也可以使用根类 CX_ROOT捕获(在不知道特定的异常类的情况下)
本文涉及到的代码
- 测试程序 ZTS_CLASS_EXCEPTION
- 对应的测试类 ZCL_TEST_EXCEPTION
- 对应的异常类 ZCX_TEST_EXCEPTION
- 通用异常类 ZCX_BC_COMM
详细代码详见文末(异常类系统不支持源码显示模式,无法复制源代码)
约定
如果你对这篇文章感兴趣,请帮忙点赞,,分享.
(如果你真的喜欢这篇文章,请记得回来打个赏,作为支持我继续下去的动力,这是一个正反馈过程. 越多的人打赏,作者越有动力分享,读者就能享受更多的福利.毕竟打赏的金额富不了我,穷不了你,却能支持这个公众号长久发文.)

公众号 : syjf1976_abap
ABAP开发技巧
申请进入公众号讨论群提问或者参与话题讨论
源代码部分
ZTS_CLASS_EXCEPTION
*&---------------------------------------------------------------------**& Report ZTS_CLASS_EXCEPTION*&---------------------------------------------------------------------**&*&---------------------------------------------------------------------*REPORT zts_class_exception.
DATA: gc_test TYPE REF TO zcl_test_exception.
START-OF-SELECTION.*创建对象 CREATE OBJECT gc_test.'返回例外:'*使用旧的例外方式: 单行代码'1' EXCEPTIONS error_1 = 1 error_2 = 2IF sy-subrc <> 0.'类方法返回例外:' && sy-subrc && '无消息'ENDIF.*使用旧的例外方式:多行代码 CALL METHOD gc_test->test_old EXPORTING'2' EXCEPTIONS123.IF sy-subrc <> 0.'类方法返回例外:' && sy-subrc && '成功消息:'ENDIF.
*使用旧的例外方式:新语法创建对象并调用方法.NEW zcl_test_exception( )->test_old( EXPORTING iv_error = '3' EXCEPTIONS error_1 = 1 error_2 = 2 error_3 = 3IF sy-subrc <> 0.'类方法返回例外:' && sy-subrc && '错误消息:'ENDIF.
*使用新的例外方法'捕获例外,错误方式:'TRY.'1'CATCH cx_root INTO DATA(lo_ref). "也可以捕获这个所有错误类的根类 DATA(lv_msg) = lo_ref->get_text( ). cl_demo_output=>write( '捕获类中的例外:' && lv_msg ). ENDTRY.*带有消息的例外. TRY. CALL METHOD gc_test->test_new( EXPORTING iv_error = '2' ). CATCH cx_root INTO lo_ref. "也可以捕获这个所有错误类的根类 lv_msg = lo_ref->get_text( ).'捕获类中的例外:' ENDTRY.
*使用新的例外方法'捕获例外,正确方式:'TRY.'1'CATCH zcx_test_exception INTO lo_ref. "可以捕获特定的异常类 lv_msg = lo_ref->get_text( ). cl_demo_output=>write( '捕获类中的例外:' && lv_msg ). ENDTRY.*带有消息的例外. TRY. CALL METHOD gc_test->test_new_right( EXPORTING iv_error = '2' ). CATCH cx_root INTO DATA(lo_ref_root). "也可以捕获这个所有错误类的根类 lv_msg = lo_ref_root->get_text( ).'捕获类中的例外:' ENDTRY.
*使用通用异常类'捕获例外,正确方式(通用异常类ZCX_BC_COMM):'TRY.'1'CATCH zcx_bc_comm INTO DATA(lo_ref_comm). "可以捕获特定的异常类 lv_msg = lo_ref_comm->get_text( ). cl_demo_output=>write( '捕获类中的例外:' && lv_msg ). ENDTRY.*带有消息的例外. TRY. CALL METHOD gc_test->test_comm( EXPORTING iv_error = '2' ). CATCH cx_root INTO DATA(lo_ref_comm_root). "也可以捕获这个所有异常类的根类 lv_msg = lo_ref_comm_root->get_text( ).'捕获类中的例外:' ENDTRY. cl_demo_output=>display( ).
ZCL_TEST_EXCEPTION
class ZCL_TEST_EXCEPTION definition public finalcreate public
public
CONSTRUCTOR methods TEST_OLD importingtypeexceptions ERROR_1 ERROR_2 ERROR_3 . methods TEST_NEW importingtype methods TEST_NEW_RIGHT importingtype raising ZCX_TEST_EXCEPTION . methods TEST_COMM importingtype raising ZCX_BC_COMM .protected section.privateENDCLASS.
CLASS
* <SIGNATURE>---------------------------------------------------------------------------------------+* | Instance Public Method ZCL_TEST_EXCEPTION->CONSTRUCTOR* +-------------------------------------------------------------------------------------------------+* +--------------------------------------------------------------------------------------</SIGNATURE> method CONSTRUCTOR. endmethod.
* <SIGNATURE>---------------------------------------------------------------------------------------+* | Instance Public* +-------------------------------------------------------------------------------------------------+* | [--->] IV_ERROR TYPE CLIKE* | [!CX!] ZCX_BC_COMM* +--------------------------------------------------------------------------------------</SIGNATURE> METHOD test_comm.CASEWHEN '1'. "测试文本 DATA:lv_str TYPE string. lv_str = '测试文本消息内容'. RAISE EXCEPTION TYPE zcx_bc_comm EXPORTING error_text = lv_str.
WHEN '2'. "测试消息DATA: ls_syst TYPE'E'.'00'.'001'.'测试SYST消息'.RAISE EXCEPTION TYPE EXPORTING syst_at_raise = ls_syst. ENDCASE. ENDMETHOD.
* <SIGNATURE>---------------------------------------------------------------------------------------+* | Instance Public* +-------------------------------------------------------------------------------------------------+* | [--->] IV_ERROR TYPE CLIKE* +--------------------------------------------------------------------------------------</SIGNATURE> METHOD test_new.DATA: lv_textid TYPECASEWHEN '1'.RAISE EXCEPTION TYPEWHEN '2'.'00'.'001'.'测试异常类消息'.* LV_TEXTID-ATTR2* LV_TEXTID-ATTR3* LV_TEXTID-ATTR4RAISE EXCEPTION TYPE EXPORTING textid = lv_textid* previous = . ENDCASE. ENDMETHOD.
* <SIGNATURE>---------------------------------------------------------------------------------------+* | Instance Public* +-------------------------------------------------------------------------------------------------+* | [--->] IV_ERROR TYPE CLIKE* | [!CX!] ZCX_TEST_EXCEPTION* +--------------------------------------------------------------------------------------</SIGNATURE> METHOD test_new_right.DATA: lv_textid TYPECASEWHEN '1'.RAISE EXCEPTION TYPEWHEN '2'.'00'.'001'.'IF_T100_DYN_MSG~MSGV1'. "ATTR1放异常类的属性. lv_textid-attr2 = '/把文本放入属性'. "文本放入属性, 最终显示的文本会前后添加 &符号* lv_textid-attr3 = '3'.* lv_textid-attr4 = '4'.RAISE EXCEPTION TYPE EXPORTING textid = lv_textid* previous = .
ENDCASE. ENDMETHOD.
* <SIGNATURE>---------------------------------------------------------------------------------------+* | Instance Public* +-------------------------------------------------------------------------------------------------+* | [--->] IV_ERROR TYPE CLIKE* | [EXC!] ERROR_1* | [EXC!] ERROR_2* | [EXC!] ERROR_3* +--------------------------------------------------------------------------------------</SIGNATURE> method TEST_OLD.CASEWHEN '1'.RAISEWHEN '2'.00) WITH '测试消息S'WHEN '3'.00) WITH '测试消息E' RAISING ERROR_3. "与函数一致, 错误消息只会触发异常.不会终止类执行 ENDCASE. endmethod.ENDCLASS.