/******************************************************************************* * GenPlay, Einstein Genome Analyzer * Copyright (C) 2009, 2014 Albert Einstein College of Medicine * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. * Authors: Julien Lajugie <julien.lajugie@einstein.yu.edu> * Nicolas Fourel <nicolas.fourel@einstein.yu.edu> * Eric Bouhassira <eric.bouhassira@einstein.yu.edu> * * Website: <http://genplay.einstein.yu.edu> ******************************************************************************/ package edu.yu.einstein.genplay.util; import java.lang.reflect.Array; /** * This class is highly inspired on the class made by Joshua Bloch (Effective Java). * Comments and reformatting changes have been made. * * * Collected methods which allow easy implementation of <code>hashCode</code>. * * Example use case: * <pre> * public int hashCode(){ * int result = HashCodeUtil.SEED; * //collect the contributions of various fields * result = HashCodeUtil.hash(result, fPrimitive); * result = HashCodeUtil.hash(result, fObject); * result = HashCodeUtil.hash(result, fArray); * return result; * } * </pre> */ public final class HashCodeUtil { private static final int fODD_PRIME_NUMBER = 37; /** * An initial value for a <code>hashCode</code>, to which is added contributions * from fields. Using a non-zero value decreases collisons of <code>hashCode</code> * values. */ public static final int SEED = 23; /** * Hashes a boolean. * @param aSeed the seed * @param aBoolean a boolean * @return the hash of the boolean */ public static int hash( int aSeed, boolean aBoolean ) { return firstTerm( aSeed ) + ( aBoolean ? 1 : 0 ); } /** * Hashes a char. * @param aSeed the seed * @param aChar a char * @return the hash of the char */ public static int hash( int aSeed, char aChar ) { return firstTerm( aSeed ) + aChar; } /** * Hashes a int. * byte and short are handled by this method, through implicit conversion. * @param aSeed the seed * @param aInt a int * @return the hash of the int */ public static int hash( int aSeed , int aInt ) { return firstTerm( aSeed ) + aInt; } /** * Hashes a long. * @param aSeed the seed * @param aLong a long * @return the hash of the long */ public static int hash( int aSeed , long aLong ) { return firstTerm(aSeed) + (int)( aLong ^ (aLong >>> 32) ); } /** * Hashes a float. * @param aSeed the seed * @param aFloat a float * @return the hash of the float */ public static int hash( int aSeed , float aFloat ) { return hash( aSeed, Float.floatToIntBits(aFloat) ); } /** * Hashes a double. * @param aSeed the seed * @param aDouble a double * @return the hash of the double */ public static int hash( int aSeed , double aDouble ) { return hash( aSeed, Double.doubleToLongBits(aDouble) ); } /** * Hashes an Object. * * <code>aObject</code> is a possibly-null object field, and possibly an array. * * If <code>aObject</code> is an array, then each element may be a primitive * or a possibly-null object. * @param aSeed the seed * @param aObject an object * @return the hash of the object */ public static int hash( int aSeed , Object aObject ) { int result = aSeed; if ( aObject == null) { result = hash(result, 0); } else if ( ! isArray(aObject) ) { result = hash(result, aObject.hashCode()); } else { int length = Array.getLength(aObject); for ( int idx = 0; idx < length; ++idx ) { Object item = Array.get(aObject, idx); //recursive call! result = hash(result, item); } } return result; } private static int firstTerm( int aSeed ){ return fODD_PRIME_NUMBER * aSeed; } private static boolean isArray(Object aObject){ return aObject.getClass().isArray(); } }