There is need to compare two objects based on class they implement? When to compare using getClass()
and when getClass().getName()
?
Is there any differ
Is there any difference between this approaches to compare two Objects class types (names)?
Yes. Two classes may have the same name if they are loaded by different ClassLoader
s.
"The basics of Java class loaders" says
At its simplest, a class loader creates a flat name space of class bodies that are referenced by a string name.
"Eclipse - a tale of two VMs (and many classloaders)" says
That means it's possible to have two classes with the same name loaded into a VM at once, provided that they have two separate ClassLoaders
When to compare using
getClass()
and whengetClass().getName()
?
If you want to know whether two objects are of the same type you should use the equals
method to compare the two classes -- the first option.
I can't imagine why you'd want to do this, but if you want to know whether two objects with different concrete types have types with the same fully qualified name, then you could use the second. If you don't understand "concrete types" and "fully qualified names" in the context of Java then you're not writing type analysis code for java so you don't want to.
As a matter of fact comparing trusted class by name is a weakness. Refer https://cwe.mitre.org/data/definitions/486.html
if (nextMonster.getClass() == monster.getClass())
is the correct way to compare classes
if (nextMonster.getClass().equals(monster.getClass()))
should still work for the same reason @Bohemian mentioned
You can use the inbuilt method of Class :
if(nextMonster.getClass().isAssignableFrom(monster.getClass()))
I ran into a problem comparing two classes using .equals. The above provided solution is not entirely accurate. Class does not implement Comparable.
Class references are not necessarily true singletons within a JVM because you can have multiple ClassLoaders.
I was writing a Maven plugin that was digging annotations out of beans after compile. The plugin had one classloader and I had my own classloader. When comparing two classes of the same name from different loaders the comparison would fail.
The implementation of Object.equals looks like this:
public boolean More ...equals(Object obj) {
return (this == obj);
}
So you will be comparing references.
If you are comparing classes and you know for sure there will only be one classloader involved you can safely use .equals or c1 == c2 but if you are not sure you should compare by name:
if(c1.getName().equals(c2.getName()) {
...
}
if you have an array to compare then use below
array.getClass().getComponentType() == X.class
Use class.equals()
:
if (nextMonster.getClass().equals(monster.getClass()))
or, because each class is like a singleton - there's only one instance of each Class per class loader, and most JVMs only have the one class loader - you can even use an identity comparison:
if (nextMonster.getClass() == monster.getClass())