 
                            Wiki Markup 
It could be necessary to check whether a given object has a specific class type or whether two objects have the same class type associated with them, for example, when implementing the equals() method. If the comparison is performed incorrectly, the code could assume that the two objects are of the same class when they are not. As a result, class names must not be compared.
...
Conversely, the assumption that two classes deriving from the same code base codebase are themselves the same is error - prone. While Although this assumption is commonly observed to be true in desktop applications, it is typically not the case with J2EE servlet containers. The containers can use different class loader instances to deploy and recall applications at runtime , without having to restart the JVM. In such situations, two objects whose classes come from the same code base codebase could appear to the JVM to be two different classes. Also note that the equals() method might not return true when comparing objects originating from the same code basecodebase.
Noncompliant Code Example
This noncompliant code example compares the name of the class (Auth) of object auth to the string "com.application.auth.DefaultAuthenticationHandler" and branches on the result of the comparison.:
| Code Block | ||
|---|---|---|
| 
 | ||
| // Determine whether object auth has required/expected class nameobject if (auth.getClass().getName().equals( "com.application.auth.DefaultAuthenticationHandler")) { // ... } | 
Comparing fully qualified class names is insufficient because distinct class loaders can load differing classes with identical fully qualified names into a single JVM.
Compliant Solution
This compliant solution compares the class object of class Auth to object auth to the class object of the class that the current class loader loads, instead of comparing just the class names.for the canonical default authentication handler:
| Code Block | ||
|---|---|---|
| 
 | ||
| // Determine whether object auth has required/expected class objectname if (auth.getClass() == this.getClass().getClassLoader().loadClass("com.application.auth.DefaultAuthenticationHandler".class)) { // ... } | 
The call to loadClass() returns the class with the specified name in the current name space (consisting of the class name and the defining class loader), and right-hand side of the comparison directly names the class of the canonical authentication handler. In the event that the canonical authentication handler had not yet been loaded, the Java runtime manages the process of loading the class. Finally, the comparison is correctly performed on the two class objects.
...
This noncompliant code example compares the names of the class objects of x and y using the equals() method. Again, it is possible that x and y are distinct classes with the same name , if they come from different class loaders.
| Code Block | ||
|---|---|---|
| 
 | ||
| // Determine whether objects x and y have the same class name if (x.getClass().getName().equals(y.getClass().getName())) { // Code assumes that the objects Objects have the same class } | 
Compliant Solution
This compliant solution correctly compares the two objects' classes.:
| Code Block | ||
|---|---|---|
| 
 | ||
| // Determine whether objects x and y have the same class if (x.getClass() == y.getClass()) { // Code assumes that the objectsObjects have the same class } | 
Risk Assessment
Comparing classes solely using their names can allow a malicious class to bypass security checks and gain access to protected resources.
| Rule | Severity | Likelihood | Detectable | 
|---|
| Repairable | Priority | Level | 
|---|---|---|
| OBJ09-J | 
| High | Unlikely | 
| Yes | 
| No | 
P9
| P6 | L2 | 
Automated Detection
| Tool | Version | Checker | Description | ||||||
|---|---|---|---|---|---|---|---|---|---|
| The Checker Framework | 
 | Signature String Checker | Ensure that the string representation of a type is properly used for example in Class.forName (see Chapter 13) | ||||||
| Klocwork | 
 | CMP.CLASS | |||||||
| Parasoft Jtest | 
 | CERT.OBJ09.CMP | Do not compare Class objects by name | ||||||
| PVS-Studio | 
 | V6054 | |||||||
| SonarQube | 
 | S1872 | Classes should not be compared by name | 
Related Guidelines
Bibliography
| [ | 
| ] | 
| Internals of Java Class Loading | 
]]></ac:plain-text-body></ac:structured-macro>
| ] | 
<ac:structured-macro ac:name="unmigrated-wiki-markup" ac:schema-version="1" ac:macro-id="3587ff37-15bc-427b-b342-5ddac1ba5fb6"><ac:plain-text-body><![CDATA[
http://java.sun.com/docs/books/jvms/second_edition/html/Concepts.doc.html]
]]></ac:plain-text-body></ac:structured-macro>
| ] | "Twelve | 
]]></ac:plain-text-body></ac:structured-macro>
<ac:structured-macro ac:name="unmigrated-wiki-markup" ac:schema-version="1" ac:macro-id="e20ff9b2-85a3-479f-ad14-d039af33e54b"><ac:plain-text-body><![CDATA[
[[Wheeler 2003
AA. Bibliography#Wheeler 03]]
[Java
http://www.dwheeler.com/secure-programs/Secure-Programs-HOWTO/java.html] Secure programming for Linux and Unix HOWTO
]]></ac:plain-text-body></ac:structured-macro>
| Rules for Developing More Secure Java Code" | |
...
OBJ08-J. Do not expose sensitive private members of an outer class from within a nested class 04. Object Orientation (OBJ) OBJ10-J. Do not use public static nonfinal variables