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.
And finally I got inspiration from Alexandru Constantin Bledea’s github.
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<T extends Exception> 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<RuntimeException>()
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.
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.
No comments:
Post a Comment