1. 程式人生 > >利用java反射比較兩個實體有哪些屬性值不一樣

利用java反射比較兩個實體有哪些屬性值不一樣

分享一個利用反射實現比較兩個實體屬性值的方法:

package net.zwq1105.test;

import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Method;
import java.sql.Timestamp;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;

/**
 * 類說明:比較兩個實體的有哪些屬性值不同
 * @author zwq1105
 * @version 建立時間:2017-3-4 下午16:30:36
 */
public class TestCompareFields {
	
	public static void main(String[] args) {
		Student s1 = new Student("123", "張三", 22, "深圳");
		Student s2 = new Student("abc", "李四", 23, "深圳");
		// 比較s1和s2不同的屬性值,其中id忽略比較
		Map<String, List<Object>> compareResult = compareFields(s1, s2, new String[]{"id"});
		System.out.println("s1和s2共有" + compareResult.size() + "個屬性值不同(不包括id)");
		System.out.println("其中:");
		Set<String> keySet = compareResult.keySet();
		for(String key : keySet){
			List<Object> list = compareResult.get(key);
			System.out.println(">>>  s1的" + key + "為" + list.get(0) + ",s2的" + key + "為" + list.get(1));
		}
	}
	
	/**
	 * 比較兩個實體屬性值,返回一個map以有差異的屬性名為key,value為一個list分別存obj1,obj2此屬性名的值
	 * @param obj1 進行屬性比較的物件1
	 * @param obj2 進行屬性比較的物件2
	 * @param ignoreArr 選擇忽略比較的屬性陣列
	 * @return 屬性差異比較結果map
	 */
	@SuppressWarnings("rawtypes")
	public static Map<String, List<Object>> compareFields(Object obj1, Object obj2, String[] ignoreArr) {
		try{
			Map<String, List<Object>> map = new HashMap<String, List<Object>>();
			List<String> ignoreList = null;
			if(ignoreArr != null && ignoreArr.length > 0){
				// array轉化為list
				ignoreList = Arrays.asList(ignoreArr);
			}
			if (obj1.getClass() == obj2.getClass()) {// 只有兩個物件都是同一型別的才有可比性
				Class clazz = obj1.getClass();
				// 獲取object的屬性描述
				PropertyDescriptor[] pds = Introspector.getBeanInfo(clazz,
						Object.class).getPropertyDescriptors();
				for (PropertyDescriptor pd : pds) {// 這裡就是所有的屬性了
					String name = pd.getName();// 屬性名
					if(ignoreList != null && ignoreList.contains(name)){// 如果當前屬性選擇忽略比較,跳到下一次迴圈
						continue;
					}
					Method readMethod = pd.getReadMethod();// get方法
					// 在obj1上呼叫get方法等同於獲得obj1的屬性值
					Object o1 = readMethod.invoke(obj1);
					// 在obj2上呼叫get方法等同於獲得obj2的屬性值
					Object o2 = readMethod.invoke(obj2);
					if(o1 instanceof Timestamp){
						o1 = new Date(((Timestamp) o1).getTime());
					}
					if(o2 instanceof Timestamp){
						o2 = new Date(((Timestamp) o2).getTime());
					}
					if(o1 == null && o2 == null){
						continue;
					}else if(o1 == null && o2 != null){
						List<Object> list = new ArrayList<Object>();
						list.add(o1);
						list.add(o2);
						map.put(name, list);
						continue;
					}
					if (!o1.equals(o2)) {// 比較這兩個值是否相等,不等就可以放入map了
						List<Object> list = new ArrayList<Object>();
						list.add(o1);
						list.add(o2);
						map.put(name, list);
					}
				}
			}
			return map;
		}catch(Exception e){
			e.printStackTrace();
			return null;
		}
	}
	
}

/**
 * 學生orm
 * @author zwq1105
 */
class Student{
	private String id;
	private String name;
	private int age;
	private String city;
	
	public Student(String id, String name, int age, String city) {
		this.id = id;
		this.name = name;
		this.age = age;
		this.city = city;
	}
	
	public String getId() {
		return id;
	}
	
	public void setId(String id) {
		this.id = id;
	}
	
	public String getName() {
		return name;
	}
	
	public void setName(String name) {
		this.name = name;
	}
	
	public int getAge() {
		return age;
	}
	
	public void setAge(int age) {
		this.age = age;
	}
	
	public String getCity() {
		return city;
	}
	
	public void setCity(String city) {
		this.city = city;
	}
	
}

執行結果:


某些型別的比較可能還存在問題,歡迎指出問題相互討論。