"should return different hash codes for non equal objects" with 0.7.5
ade90036 opened this issue · 2 comments
I have a POJO and using intelliJ i've created toString, equals, hashCode methods.
package com.arizzini.api.model;
import com.arizzini.api.runner.Response;
import com.arizzini.api.core.model.Environment;
import java.io.Serializable;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* Created by arizzini on 11/2/16.
*/
public class ServiceObject implements Serializable {
public final String version;
public final String title;
public final String artifactId;
public final String host;
public final String context;
public final Environment environment;
public final List<OperationObject> operationObjects;
public final boolean hasOrdering;
public Integer elapseTimeSeconds;
public Date lastRun;
public Map<String,Response> responseMap;
public ServiceObject(String artifactId, String title, String version, String host, String context, Environment environment, List<OperationObject> operationObjects, boolean hasOrdering) {
this.version = version;
this.title = title;
this.artifactId = artifactId;
this.host = host;
this.context = context;
this.environment = environment;
this.operationObjects = operationObjects;
this.responseMap = new HashMap<>();
this.hasOrdering = hasOrdering;
}
public void clearResponseMap() {
responseMap.clear();
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
ServiceObject that = (ServiceObject) o;
if (hasOrdering != that.hasOrdering) return false;
if (version != null ? !version.equals(that.version) : that.version != null) return false;
if (title != null ? !title.equals(that.title) : that.title != null) return false;
if (artifactId != null ? !artifactId.equals(that.artifactId) : that.artifactId != null) return false;
if (host != null ? !host.equals(that.host) : that.host != null) return false;
if (context != null ? !context.equals(that.context) : that.context != null) return false;
if (environment != that.environment) return false;
return true;
}
@Override
public int hashCode() {
int result = version != null ? version.hashCode() : 0;
result = 31 * result + (title != null ? title.hashCode() : 0);
result = 31 * result + (artifactId != null ? artifactId.hashCode() : 0);
result = 31 * result + (host != null ? host.hashCode() : 0);
result = 31 * result + (context != null ? context.hashCode() : 0);
result = 31 * result + (environment != null ? environment.hashCode() : 0);
result = 31 * result + (operationObjects != null ? operationObjects.hashCode() : 0);
result = 31 * result + (hasOrdering ? 1 : 0);
return result;
}
@Override
public String toString() {
final StringBuffer sb = new StringBuffer("ServiceObject{");
sb.append("version='").append(version).append('\'');
sb.append(", title='").append(title).append('\'');
sb.append(", artifactId='").append(artifactId).append('\'');
sb.append(", host='").append(host).append('\'');
sb.append(", context='").append(context).append('\'');
sb.append(", environment=").append(environment);
sb.append(", operationObjects=").append(operationObjects);
sb.append(", hasOrdering=").append(hasOrdering);
sb.append('}');
return sb.toString();
}
}
When i run the test i'm getting the following:
Class com.arizzini.api.model.ServiceObject has bad 'hashCode' method implementation.
The hashCode method should return different hash codes for non equal objects.
Current implementation returns same hash codes.
Object:
ServiceObject{version='www.pojo.pl', title='www.pojo.pl', artifactId='www.pojo.pl', host='www.pojo.pl', context='www.pojo.pl', environment=PRODUCTION, operationObjects=[], hasOrdering=true}
and
ServiceObject{version='www.pojo.pl', title='www.pojo.pl', artifactId='www.pojo.pl', host='www.pojo.pl', context='www.pojo.pl', environment=PRODUCTION, operationObjects=[], hasOrdering=true}
should have different hash codes:
-923710817
and
-923710817
And the test looks like this:
@Test
public void ServiceObjectPojoOperations() throws Exception {
// given
final Class<?> classUnderTest = ServiceObject.class;
// when
// then
assertPojoMethodsFor(classUnderTest)
.testing(Method.CONSTRUCTOR)
.testing(Method.EQUALS)
.testing(Method.HASH_CODE)
.testing(Method.TO_STRING)
.areWellImplemented();
}
Is there a way i can debug what is going on?
The error message is a bit confusing...
The test failure is clearly stated here: "The hashCode method should return different hash codes for non equal objects.", however, when the two objects are printed in the following statements they appear to be identical??
Should it be printing the actual objects which is trying to generate the hashCode instead?
Any ideas how i could fix this?
I'm not using Date, (in the hashCode).
BTW: i 0.7.5 is not in maven central so i had manually include the jar in my project lib folder.
Regards
Hi, thanks for reporting an issue ;)
First of all, pojo-tester is published only to bintray, so you can use it as dependency in your build - just define jcenter as repository as described in here http://stackoverflow.com/questions/31961025/using-jcenter-and-mavencentral-both-in-android-studio.
Second, you can debug by enabling DEBUG
level on your logger when you perform tests. See more at http://www.pojo.pl/writing-tests/#debugging.
Third, at first glance I can spy with my little eye, that the assertion exception is good because:
-
You tell to pojo-tester: "test this class changing all of its fields". So it changes all of them including:
elapseTimeSeconds
,lastRun
,responseMap
. And you didn't implement them intoString()
method, so you can't see in log, that they are actually different - they can have different e.g.lastRun
field.
If you don't want to test those fields, just exclude them as described here http://www.pojo.pl/writing-tests/#choosing-fields -
Your
equals
andhashcode
methods seem to be wrong implemented because you includedoperationObjects
inequals
implementation but forgot / didn't include it inhashcode
implementation.
Let me know if this answer is satisfactory.