
org.tinygroup.commons.tools.EqualsUtil Maven / Gradle / Ivy
/**
* Copyright (c) 1997-2013, www.tinygroup.org ([email protected]).
*
* Licensed under the GPL, Version 3.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.gnu.org/licenses/gpl.html
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.tinygroup.commons.tools;
import org.apache.commons.lang.builder.EqualsBuilder;
import java.lang.reflect.AccessibleObject;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
/**
*
* 功能说明: 对象比较工具方法
* 开发人员: renhui
* 开发时间: 2014-2-24
*/
public class EqualsUtil {
public static boolean reflectionEquals(Object lhs, Object rhs) {
return EqualsBuilder.reflectionEquals(lhs, rhs);
}
public static boolean reflectionEquals(Object lhs, Object rhs, boolean testTransients) {
return EqualsBuilder.reflectionEquals(lhs, rhs, testTransients);
}
public static boolean reflectionEquals(Object lhs, Object rhs, boolean testTransients, Class reflectUpToClass) {
return EqualsBuilder.reflectionEquals(lhs, rhs, testTransients, reflectUpToClass);
}
public static boolean reflectionEquals(Object lhs, Object rhs, boolean testTransients, Class reflectUpToClass,
String[] excludeFields) {
return EqualsBuilder.reflectionEquals(lhs, rhs, testTransients, reflectUpToClass, excludeFields);
}
public static boolean reflectionEquals(Object lhs, Object rhs, String[] excludeFields) {
return EqualsBuilder.reflectionEquals(lhs, rhs, excludeFields);
}
public static boolean reflectionEquals(Object lhs, Object rhs, Collection /*String*/ excludeFields) {
return EqualsBuilder.reflectionEquals(lhs, rhs, excludeFields);
}
public static boolean reflectionCompareEquals(Object lhs, Object rhs,String[] compareFields){
return reflectionCompareEquals(lhs, rhs, false, null, compareFields);
}
public static boolean reflectionCompareEquals(Object lhs, Object rhs,Collection compareFields){
return reflectionCompareEquals(lhs, rhs, false, null, CollectionUtil.toNoNullStringArray(compareFields));
}
public static boolean reflectionCompareEquals(Object lhs, Object rhs,boolean testTransients,Class reflectUpToClass,String[] compareFields) {
if (lhs == rhs) {
return true;
}
if (lhs == null || rhs == null) {
return false;
}
// Find the leaf class since there may be transients in the leaf
// class or in classes between the leaf and root.
// If we are not testing transients or a subclass has no ivars,
// then a subclass can test equals to a superclass.
Class lhsClass = lhs.getClass();
Class rhsClass = rhs.getClass();
Class testClass;
if (lhsClass.isInstance(rhs)) {
testClass = lhsClass;
if (!rhsClass.isInstance(lhs)) {
// rhsClass is a subclass of lhsClass
testClass = rhsClass;
}
} else if (rhsClass.isInstance(lhs)) {
testClass = rhsClass;
if (!lhsClass.isInstance(rhs)) {
// lhsClass is a subclass of rhsClass
testClass = lhsClass;
}
} else {
// The two classes are not related.
return false;
}
EqualsBuilder equalsBuilder = new EqualsBuilder();
try {
reflectionCompareAppend(lhs, rhs, testClass, equalsBuilder, testTransients, compareFields);
while (testClass.getSuperclass() != null && testClass != reflectUpToClass) {
testClass = testClass.getSuperclass();
reflectionCompareAppend(lhs, rhs, testClass, equalsBuilder, testTransients, compareFields);
}
} catch (IllegalArgumentException e) {
// In this case, we tried to test a subclass vs. a superclass and
// the subclass has ivars or the ivars are transient and
// we are testing transients.
// If a subclass has ivars that we are trying to test them, we get an
// exception and we know that the objects are not equal.
return false;
}
return equalsBuilder.isEquals();
}
/**
* Appends the fields and values defined by the given object of the
* given Class.
*
* @param lhs the left hand object
* @param rhs the right hand object
* @param clazz the class to append details of
* @param builder the builder to append to
* @param useTransients whether to test transient fields
* @param excludeFields array of field names to exclude from testing
*/
private static void reflectionCompareAppend(
Object lhs,
Object rhs,
Class clazz,
EqualsBuilder builder,
boolean useTransients,
String[] compareFields) {
Field[] fields = clazz.getDeclaredFields();
List compareFieldList = compareFields != null ? Arrays.asList(compareFields) : Collections.EMPTY_LIST;
AccessibleObject.setAccessible(fields, true);
for (int i = 0; i < fields.length && builder.isEquals(); i++) {
Field f = fields[i];
if (compareFieldList.contains(f.getName())
&& (f.getName().indexOf('$') == -1)
&& (useTransients || !Modifier.isTransient(f.getModifiers()))
&& (!Modifier.isStatic(f.getModifiers()))) {
try {
builder.append(f.get(lhs), f.get(rhs));
} catch (IllegalAccessException e) {
//this can't happen. Would get a Security exception instead
//throw a runtime exception in case the impossible happens.
throw new InternalError("Unexpected IllegalAccessException");
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy