首页 > 编程语言 > 详细

Java和SAP ABAP的异常处理

时间:2020-08-26 22:33:58      阅读:120      评论:0      收藏:0      [点我收藏+]

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:

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:

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:

  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:

  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.

Further reading

I have written a series of blogs which compare the language feature among ABAP, JavaScript and Java. You can find a list of them below:

要获取更多Jerry的原创文章,请关注公众号"汪子熙":
技术分享图片

Java和SAP ABAP的异常处理

原文:https://www.cnblogs.com/sap-jerry/p/13568365.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!