前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java和SAP ABAP的异常处理

Java和SAP ABAP的异常处理

作者头像
Jerry Wang
发布2020-08-25 10:00:16
7040
发布2020-08-25 10:00:16
举报

Recently I am prepare an internal training and have been racking my brains to find a real example for my attendees about writting a “correct” program which gets rejected by compiler. The tricky point here is as a programmer, we always treat compiler as our god: if compiler complains that our program has errors, then we are wrong. Programmers tend to believe in that compiler will NEVER make mistakes.

Checked and unchecked exception in Java

Let’s see the following Java code:

代码语言:javascript
复制
package exception;
import java.sql.SQLException;
public class ExceptionForQuiz<T extends Exception> {
	private void pleaseThrow(final Exception t) throws T {
		throw (T) t;
	}
	public static void main(final String[] args) {
		try {
			new ExceptionForQuiz<RuntimeException>().pleaseThrow(new SQLException());
		}
		catch( final SQLException ex){
			System.out.println("Jerry print");
			ex.printStackTrace();
		}
	}
}

What result this program will generate? Let’s analyze it step by step.

(1) The class ExceptionForQuiz uses a generic typing syntax extends to declare a bound that T only accepts Exception and its sub classes. As a result in my main method code the creation of new ExceptionForQuiz via the below code is legal since according to JDK source code, RuntimeException is subclass of Exception.

new ExceptionForQuiz()

Also keep in mind that RuntimeException is a kind of Unchecked exception ( do not need to be declared in method where it might be raised ), which will be compared with ABAP exception later.

(2) According to Java Document, the type parameter in generic type declaration will be replaced by its bound during compile, in my exception RuntimeException will be replaced by Exception. This is called Type Erasure. As a result, let’s forget about the try – catch for the moment. This is original code:

This is the code decompiled from ExceptionForQuiz.class:

You can observe the fact of type erasure clearly. You can also check the byte code by command javap, where the RuntimeException is erased.

(3) Now let’s see the result of this quiz. The correct answer is: this program cannot pass compile! Compiler considers that SQLException could never have possibility to be raised from within TRY block.

Unfortunately, this is what I would like to do: raise SQLException via method pleaseThrow and catch it in catch block.

How to achieve my requirement?

Just change one line as highlighted below. Instead of catching SQLException, I now catch RuntimeException in order to pacify the compiler.

Now there is no compilation error any more but when executing it, I find the raised SQLException still cannot be caught as I expect. My println is not executed at all.

I have to catch the Exception, the super class of all other exception instead ( like CX_ROOT in ABAP ), which is not a good practice in exception handling area.

Handleable and Unhandleable Exception in ABAP

You can find both definition in ABAP help.

Let’s now do the similar exercise as we did previous in Java. Create a method with below signature.

Now make the first test:

代码语言:javascript
复制
DATA(lo_test) = NEW zcl_exception_test( ).

  DATA: lo_exception TYPE REF TO cx_atd_exception.

  CREATE OBJECT lo_exception.

  WRITE:/ 'First test' COLOR COL_NEGATIVE.
  TRY.
      lo_test->please_throw( lo_exception ).
    CATCH cx_atd_exception INTO DATA(exception1).
      WRITE:/ 'Jerry: ' , exception1->get_text( ).
  ENDTRY.

It works as expected, the raised exception is caught.

Now the second test:

代码语言:javascript
复制
  DATA: lo_exception2 TYPE REF TO cx_sql_exception.

  CREATE OBJECT lo_exception2.

  WRITE:/ 'Second test' COLOR COL_NEGATIVE.

  TRY.
      lo_test->please_throw( lo_exception2 ).
    CATCH cx_sql_exception INTO DATA(exception).
      WRITE:/ 'In catch sql exception:' , exception->get_text( ).
  ENDTRY.

The code is exactly the same as the first test, except that the exception is changed from CX_ATD_EXCEPTION to CX_SQL_EXCEPTION. And result for the second test, 囧 …

In order to make this CX_SQL_EXCEPTION caught-able, I have to write the same dirty code as we did in Java example:

Although this time it works, but what is the reason of the different behaviors of these two examples?

The error message and short dump description have already given us a hint. CX_ATD_EXCEPTION’s super class: CX_NO_CHECK. As its description says, it is not necessary to manually declare it in method signature using RAISING keyword.

And CX_SQL_EXCEPTION’s super class: CX_STATIC_CHECK

As a result now we have another solution: Create another version of PLEASE_THROW method with RAISING keyword:

Use this new version and now CX_SQL_EXCEPTION could be caught:

代码语言:javascript
复制
  WRITE:/ 'Third test' COLOR COL_NEGATIVE.
  TRY.
      lo_test->please_throw2( lo_exception2 ).
    CATCH cx_sql_exception INTO DATA(exception3).
      WRITE:/ 'In catch sql exception:' , exception3->get_text( ).
  ENDTRY.
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2020-08-23 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Checked and unchecked exception in Java
  • How to achieve my requirement?
  • Handleable and Unhandleable Exception in ABAP
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档