Sometime when looking through code, I see many methods specify an annotation:
@SuppressWarnings("unchecked")
What does this mean?
Sometime when looking through code, I see many methods specify an annotation:
@SuppressWarnings("unchecked")
What does this mean?
In Java, generics are implemented by means of type erasure. For instance, the following code.
Is compiled to the following.
And
List.of
is defined as.Which after type erasure becomes.
The compiler has no idea what are generic types at runtime, so if you write something like this.
Java Virtual Machine has no idea what generic types are while running a program, so this compiles and runs, as for Java Virtual Machine, this is a cast to
List
type (this is the only thing it can verify, so it verifies only that).But now add this line.
And JVM will throw an unexpected
ClassCastException
, as Java compiler inserted an implicit cast.An
unchecked
warning tells a programmer that a cast may cause a program to throw an exception somewhere else. Suppressing the warning with@SuppressWarnings("unchecked")
tells the compiler that the programmer believes the code to be safe and won't cause unexpected exceptions.Why would you want to do that? Java type system isn't good enough to represent all possible type usage patterns. Sometimes you may know that a cast is safe, but Java doesn't provide a way to say so - to hide warnings like this,
@SupressWarnings("unchecked")
can be used, so that a programmer can focus on real warnings. For instance,Optional.empty()
returns a singleton to avoid allocation of empty optionals that don't store a value.This cast is safe, as the value stored in an empty optional cannot be retrieved so there is no risk of unexpected class cast exceptions.
A warning by which the compiler indicates that it cannot ensure type safety. The term "unchecked" warning is misleading. It does not mean that the warning is unchecked in any way. The term "unchecked" refers to the fact that the compiler and the runtime system do not have enough type information to perform all type checks that would be necessary to ensure type safety. In this sense, certain operations are "unchecked". The most common source of "unchecked" warnings is the use of raw types. "unchecked" warnings are issued when an object is accessed through a raw type variable, because the raw type does not provide enough type information to perform all necessary type checks.
Example (of unchecked warning in conjunction with raw types):
TreeSet se t = new TreeSet(); set.add("abc"); // unchecked warning set.remove("abc");
warning: [unchecked] unchecked call to add(E) as a member of the raw type java.util.TreeSet set.add("abc");
^ When the add method is invoked the compiler does not know whether it is safe to add a String object to the collection. If the TreeSet is a collection that contains String s (or a supertype thereof), then it would be safe. But from the type information provided by the raw type TreeSet the compiler cannot tell. Hence the call is potentially unsafe and an "unchecked" warning is issued.
"unchecked" warnings are also reported when the compiler finds a cast whose target type is either a parameterized type or a type parameter.
Example (of an unchecked warning in conjunction with a cast to a parameterized type or type variable):
class Wrapper<T> { private T wrapped ; public Wrapper (T arg) {wrapped = arg;} ... p ublic Wrapper <T> clone() { Wrapper<T> clon = null; try {
clon = (Wrapper<T>) super.clone(); // unchecked warning } catch (CloneNotSupportedException e) {
throw new InternalError();
} try {
Class<?> clzz = this.wrapped.getClass(); Method meth = clzz.getMethod("clone", new Class[0]); Object dupl = meth.invoke(this.wrapped, new Object[0]); clon.wrapped = (T) dupl; // unchecked warning } catch (Exception e) {} return clon; } }
warning: [unchecked] unchecked cast found : java.lang.Object required: Wrapper <T> clon = ( Wrapper <T>)super.clone();
^ warning: [unchecked] unchecked cast found : java.lang.Object required: T clon. wrapped = (T)dupl;
A cast whose target type is either a (concrete or bounded wildcard) parameterized type or a type parameter is unsafe, if a dynamic type check at runtime is involved. At runtime, only the type erasure is available, not the exact static type that is visible in the source code. As a result, the runtime part of the cast is performed based on the type erasure, not on the exact static type. In the example, the cast to Wrapper would check whether the object returned from super.clone is a Wrapper , not whether it is a wrapper with a particular type of members. Similarly, the casts to the type parameter T are cast to type Object at runtime, and probably optimized away altogether. Due to type erasure, the runtime system is unable to perform more useful type checks at runtime.
In a way, the source code is misleading, because it suggests that a cast to the respective target type is performed, while in fact the dynamic part of the cast only checks against the type erasure of the target type. The "unchecked" warning is issued to draw the programmer's attention to this mismatch between the static and dynamic aspect of the cast.
Please refer: What is an "unchecked" warning?
You can suppress the compiler warnings and tell the generics that the code which you had written is legal according to it.
Example:
Sometimes Java generics just doesn't let you do what you want to, and you need to effectively tell the compiler that what you're doing really will be legal at execution time.
I usually find this a pain when I'm mocking a generic interface, but there are other examples too. It's usually worth trying to work out a way of avoiding the warning rather than suppressing it (the Java Generics FAQ helps here) but sometimes even if it is possible, it bends the code out of shape so much that suppressing the warning is neater. Always add an explanatory comment in that case!
The same generics FAQ has several sections on this topic, starting with "What is an "unchecked" warning?" - it's well worth a read.
It could also mean that the current Java type system version isn't good enough for your case. There were several JSR propositions / hacks to fix this: Type tokens, Super Type Tokens, Class.cast().
If you really need this supression, narrow it down as much as possible (e.g. don't put it onto the class itself or onto a long method). An example:
One trick is to create an interface that extends a generic base interface...
Then you can check it with instanceof before the cast...