Why User defined Exception not throws over Generic Exception? - java

I have created below User Defined Exception (MyException) class which extends Exception, Now I manually try to create a NullPointerException, But instead of throwing (MyException), it throws the same Generic(NullPointerException) Exception
public class Sample {
public static void main(String[] args) throws MyException {
String a = null;
callMethod(a);
}
public static void callMethod(String a) throws MyException {
a.toString();
}
}
public class MyException extends Exception {
public MyException(String exception)
{
super(exception);
}
}
Below is the Exception I'm getting
Exception in thread "main" java.lang.NullPointerException
at Sample.callMethod(Sample.java:9)
at Sample.main(Sample.java:5)
Process finished with exit code 1

The fact that you declare that a method throws an exception does not mean it will throw that exception when any other exception is thrown in the code.
In this case, a is null so the exception that's thrown is a NullPointerException. If you wanted to throw your custom exception you would have to check whether your value is null and throw your custom exception, like this
public static void callMethod(String a) throws MyException {
if (a == null) {
throw new MyException(e);
}
a.toString();
}

Declaring that a method throws MyException means that it may throw that exception.
It does not mean that all thrown exceptions automatically get wrapped in your MyException

Using throws MyException does mean that the method can throw an exception and u need to handle it, but it doesn't mean every exception u get inside the method will be thrown as MyException. If u want to throw every Exception u get as MyException you can do :
public static void main(String[] args) throws MyException {
try{
String a = null;
callMethod(a);
}catch(Exception e){
throw new MyException(...);
}
}

Related

Usage of throws and try-catch in the same method [duplicate]

This question already has an answer here:
What does "error: unreported exception <XXX>; must be caught or declared to be thrown" mean and how do I fix it?
(1 answer)
Closed 6 months ago.
Can we use throws and try-catch in the same method?
public class Main
{
static void t() throws IllegalAccessException {
try{
throw new IllegalAccessException("demo");
} catch (IllegalAccessException e){
System.out.println(e);
}
}
public static void main(String[] args){
t();
System.out.println("hello");
}
}
The error displayed is
Main.java:21: error: unreported exception IllegalAccessException; must be caught or declared to be thrown
t();
^
1 error
So I thought of modifying the code and I added an another throws statement to main() method and the rest is same.
public class Main
{static void t() throws IllegalAccessException {
try{
throw new IllegalAccessException("demo");
} catch (IllegalAccessException e){
System.out.println(e);
}
}
public static void main(String[] args) throws IllegalAccessException{
t();
System.out.println("hello");
}
}
But now I'm getting the desired output.
But I have some questions...
Can we use throws and try-catch in single method?
In my case is it necessary to add two throws statement , if not tell me the appropriate place to add?
In your code
void t() throws IllegalAccessException
you are telling the compiler that this code throws an exception (whether it does or not is another matter), and so any method calling this method either has to catch it or declare that it also throws it etc. etc.
As you are not actually throwing an exception from t you can remove the declaration.
void t()

tips "Unhandled exception type xxx" in eclipse

I don't think I understand the try-catch block and throws really.
public class TestException {
public static void main(String[] args) {
new TestException().tt();
}
public void tt() {
try {
throw new RuntimeException();
}catch (Exception e) {
throw e;
}
}
}
When in Eclipse, there is an error hint about 'Unhandled exception type xxx', and if you run this, you will get an
Exception in thread "main" java.lang.Error: Unresolved compilation problem:
Unhandled exception type Exception
But in Idea, there's no errors. It runs and throws the exception correctly.
In my opnion, the 'e' was not declared as a RuntimeException(althrough it is an RuntimeException), so the tt() method must be declared with throws. But actually it's not. Why?
This should answer your question:
public class TestException {
public static void main(String[] args) {
new TestException().tt();
}
public void tt() {
try {
throw new RuntimeException();
}catch (Exception e) {
throw new RuntimeException(e);
}
}
}
If you use throws, you tell those who use your function, "My function may throw exceptions. You have to handle that."
You should get difference of throws and throw in this sentence.
If you use try-catch, you handle that exception.
1) You should add throws keyword like below
public class TestException {
public static void main(String[] args) {
new TestException().tt();
}
public void tt() **throws Exception** {
try {
throw new RuntimeException();
}catch (Exception e) {
throw e;
}
}
}
2) Handle exception where you use function
public class TestException {
public static void main(String[] args) {
try{
new TestException().tt();
}catch(Exception e){
System.out.println("Error handled");
}
}
public void tt() throws Exception {
try {
throw new RuntimeException();
}catch (Exception e) {
throw e;
}
}
}
In general if you catch an exception you handle it. Like this
public class TestException {
public static void main(String[] args) {
new TestException().tt();
}
public void tt() {
try {
throw new RuntimeException();
}catch (Exception e) {
System.out.println("Error caught! ")
}
}
}
or
public class TestException {
public static void main(String[] args) {
try {
new TestException().tt();
}catch(Exception e){
System.out.println("Error caught! ")
}
}
public void tt() throws RuntimeException {
throw new RuntimeException();
}
}
You can also throw other's exception
public class TestException {
public static void main(String[] args) {
try{
new TestException().a();
}catch(Exception e){
System.out.println("Error handled");
}
}
public void a() throws Exception {
b();
}
public void b() throws Exception {
c();
}
public void c() throws Exception {
throw new RuntimeException();
}
}
I think that you want to look into 'Checked Exceptions' and 'Unchecked Exceptions'.
Only Checked Exceptions need to be declared in a methods signature, and RuntimeException is an unchecked exception (though you can declare it if you like - it just isn't necessary to compile).
https://docs.oracle.com/javase/7/docs/api/java/lang/Exception.html
The API for java Exception says:
"The class Exception and any subclasses that are not also subclasses of RuntimeException are checked exceptions. Checked exceptions need to be declared in a method or constructor's throws clause if they can be thrown by the execution of the method or constructor and propagate outside the method or constructor boundary"
https://docs.oracle.com/javase/7/docs/api/java/lang/RuntimeException.html
The API for java RuntimeException says:
"RuntimeException and its subclasses are unchecked exceptions. Unchecked exceptions do not need to be declared in a method or constructor's throws clause if they can be thrown by the execution of the method or constructor and propagate outside the method or constructor boundary."
In my opnion, the 'e' was not declared as a RuntimeException(althrough it is an RuntimeException), so the tt() method must be declared with throws. But actually it's not. Why?
Let's consider what we know:
When using rethrow syntax, the existing exception object (e) is rethrown.
e is an object of class Exception, or one of its subtypes.
RuntimeException is a subtype of exception, and is not compiled time checked, so it's possible the re-thrown object is a non compile time checked object.
The compiler cannot see a place where the code definitely, or even possibly throws a compile checked exception, and so it makes sense that it does not force those semantics.
For example, if you change your catch to an IOException, the compiler will not allow you to do that without a line in the try which could possibly lead to an IOException.
If you added such a line, then the compiler would recognize that the throw would rethrow a compile time checked exception, and make you catch it again, or mark the function with the appropriate throws clause.
As for eclipse, your code compiles OK in mine with my JDK.

catching java.lang.ArrayIndexOutOfBoundsException

I'm new in Java. Can anyone help me with explanation, why the catch is not catching the MyException (which extends ArrayIndexOutOfBoundsException)?
My example:
public class TestClass {
public static void main(String[] args) {
try{
doTest();
}
catch(MyException me){
System.out.println("MyException is here");
}
}
static void doTest() throws MyException{
int[] array = new int[10];
array[10] = 1000;
}
}
class MyException extends ArrayIndexOutOfBoundsException {
public MyException(String msg){
super(msg);
}
}
The result is:
"Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 10         at TestClass.doTest(TestClass.java:14)         at TestClass.main(TestClass.java:5)"
Why is not "MyException is here"?
Your method actually throws only ArrayIndexOutOfBoundsException.
You catch MyException but that is not what is being thrown, so the catch clause has no effect.
If you wanted to throw MyException you would have to modify the method to catch ArrayIndexOutOfBoundsException and throw MyException instead.
You are confusing the subtype-supertype-relationship.
The code as it is throws an ArrayIndexOutOfBoundsException but not a MyException. Catching the latter will not work because an AIOOBE is not a ME. Your ME is a subtype of AIOOBE.
On the other hand, an AIOOBE has a supertype: IndexOutOfBoundsException. If you have a catch clause for this one, you will get the desired behavior because an AIOOBE is a IOOBE.
Or you could simply throw your ME yourself: throw new MyException(...)
Your doTest method does not throw your custom exception. To throw an exception use
throw new MyException("your message");

Java unhandled Exception

I'm currently developing an Android app using Android Studio and I encountered an error that I can't understand. Here's an example of my code :
public class MyClass {
public void method1() throws MyException {
methodIO(); // Unhandled Exception: java.io.IOException
methodRun(); // OK
}
public void method2() throws Exception {
methodIO(); // OK
methodRun(); // OK
}
public void methodIO() throws IOException {}
public void methodRun() throws RuntimeException {}
}
class MyException extends Exception {
public MyException(){super();}
public MyException(String message, int errorCode) {
super(message);
}
}
I can't figure out why in method1(), I have the following error "Unhandled Exception: java.io.IOException" while method2() compiles fine. The issue doesn't appear with a method throwing a RuntimeException even if both classes are inheritd from Exception. Does someone know what's going on here ?
Ps : I'd like to avoid using a try/catch bloc or adding a new throws clause
MyException doesn't extend IOException, so the IOException thrown by methodIO isn't handled currently.
It works in method2 because Exception is a common superclass of IOException and MyException, so it specifies how both types of exception should be handled.
Add it to the throws clause of method1:
public void method1() throws MyException, IOException {
or catch in the method body:
public void method1() throws MyException {
try {
methodIO();
} catch (IOException e) {
// Handle appropriately.
}
}
method2() has a throws declaration for the base type Exception, of which IOException is a derived type; whereas method1() throws a MyException, which IOException does not derive from. Add IOException to the throws declaration of method1():
public void method1() throws MyException, IOException {
methodIO();
}
or you could always catch the IOException and throw a MyException instead:
public void method1() throws MyException {
try {
methodIO();
} catch (IOException e) {
throw new MyException();//construct MyException however; this is just an example
}
}

Java methods don't always need to declare throwing checked exceptions [duplicate]

I would expect the following code to raise a compile-time error on throw t;, because main is not declared to throw Throwable, but it compiles successfully (in Java 1.7.0_45), and produces the output you would expect it to if that compile-time error was fixed.
public class Test {
public static void main(String[] args) {
try {
throw new NullPointerException();
} catch(Throwable t) {
System.out.println("Caught "+t);
throw t;
}
}
}
It also compiles if Throwable is changed to Exception.
This does not compile, as expected:
public class Test {
public static void main(String[] args) {
try {
throw new NullPointerException();
} catch(Throwable t) {
Throwable t2 = t;
System.out.println("Caught "+t2);
throw t2;
}
}
}
This compiles:
public class Test {
public static void main(String[] args) {
try {
throwsRuntimeException();
} catch(Throwable t) {
System.out.println("Caught "+t);
throw t;
}
}
public static void throwsRuntimeException() {
throw new NullPointerException();
}
}
This does not:
public class Test {
public static void main(String[] args) {
try {
throwsCheckedException();
} catch(Throwable t) {
System.out.println("Caught "+t);
throw t;
}
}
public static void throwsCheckedException() {
throw new java.io.IOException();
}
}
This compiles as well:
public class Test {
public static void main(String[] args) throws java.io.IOException {
try {
throwsIOException();
} catch(Throwable t) {
System.out.println("Caught "+t);
throw t;
}
}
public static void throwsIOException() throws java.io.IOException {
throw new java.io.IOException();
}
}
A more complex example - the checked exception is caught by an outer catch block, instead of being declared to be thrown. This compiles:
public class Test {
public static void main(String[] args) {
try {
try {
throwsIOException();
} catch(Throwable t) {
System.out.println("Caught "+t);
throw t;
}
} catch(java.io.IOException e) {
System.out.println("Caught IOException (outer block)");
}
}
public static void throwsIOException() throws java.io.IOException {
throw new java.io.IOException();
}
}
So there seems to be a special case to allow rethrowing exceptions when the compiler can determine that the caught exception is always legal to re-throw. Is this correct? Where is this specified in the JLS? Are there any other obscure corner-cases like this?
This is covered by JLS 11.2.2 (emphasis mine):
A throw statement whose thrown expression is a final or effectively final exception parameter of a catch clause C can throw an exception class E iff:
E is an exception class that the try block of the try statement which declares C can throw; and
E is assignment compatible with any of C's catchable exception classes; and
(...)
In other words, E, the type referenced in the doc, is the type that can be thrown, not the type of the catch clause parameter that catches it (the catchable exception class). It just has to be assignment compatible to the catch clause parameter, but the parameter's type is not used in analysis.
This is why the go out of their way to say a final or effectively final exception parameter--if t in your example were reassigned, the analysis would go out the window.
Because the compiler is smart enough to know that a checked exception can not be thrown from the try block, and the caught Throwable is thus not a checked exception that must be declared.
Note that this is true since Java 7, if I'm not mistaken.
When you catch Throwable or Exception and the variable is effectively final you can rethrow the same variable and the compiler will know which checked exceptions you could have thrown in the try {} catch block.

Categories