JAvaオブジェクト間の属性値のコピー

6506 ワード

JAvaオブジェクト間の属性値レプリケーションは、多くのオープンソースフレームワークでも実装されています.ここでは、独自の実装、apache commons-beanutils、Springの3つの方法で実装されています.
1自分で実現する.反射を使用して、ソースオブジェクトgetterメソッドで属性値を取得し、ターゲットオブジェクトのsetterメソッドでターゲットオブジェクトに設定します.パフォーマンスはあまり考慮されていませんが、一括または頻繁に実行する必要がある場合は、この実装は推奨されません.次に、ソースコードを示します.
/**
	 *               
	 * @param from
	 * @param to
	 */
	public static void copyProperties(Object from, Object to) throws Exception {
		copyPropertiesExclude(from, to, null);
	}
	
	/**
	 *       
	 * @param from
	 * @param to
	 * @param excludsArray       
	 * @throws Exception
	 */
	@SuppressWarnings("unchecked")
	public static void copyPropertiesExclude(Object from, Object to, String[] excludsArray) throws Exception {
		List excludesList = null;
		if(excludsArray != null && excludsArray.length > 0) {
			excludesList = Arrays.asList(excludsArray);	//      
		}
		Method[] fromMethods = from.getClass().getDeclaredMethods();
		Method[] toMethods = to.getClass().getDeclaredMethods();
		Method fromMethod = null, toMethod = null;
		String fromMethodName = null, toMethodName = null;
		for (int i = 0; i < fromMethods.length; i++) {
			fromMethod = fromMethods[i];
			fromMethodName = fromMethod.getName();
			if (!fromMethodName.contains("get"))
				continue;
			//      
			if(excludesList != null && excludesList.contains(fromMethodName.substring(3).toLowerCase())) {
				continue;
			}
			toMethodName = "set" + fromMethodName.substring(3);
			toMethod = findMethodByName(toMethods, toMethodName);
			if (toMethod == null)
				continue;
			Object value = fromMethod.invoke(from, new Object[0]);
			if(value == null)
				continue;
			//       
			if(value instanceof Collection) {
				Collection newValue = (Collection)value;
				if(newValue.size() <= 0)
					continue;
			}
			toMethod.invoke(to, new Object[] {value});
		}
	}
	
	/**
	 *        ,           
	 * @param from
	 * @param to
	 * @param includsArray
	 * @throws Exception
	 */
	@SuppressWarnings("unchecked")
	public static void copyPropertiesInclude(Object from, Object to, String[] includsArray) throws Exception {
		List includesList = null;
		if(includsArray != null && includsArray.length > 0) {
			includesList = Arrays.asList(includsArray);	//      
		} else {
			return;
		}
		Method[] fromMethods = from.getClass().getDeclaredMethods();
		Method[] toMethods = to.getClass().getDeclaredMethods();
		Method fromMethod = null, toMethod = null;
		String fromMethodName = null, toMethodName = null;
		for (int i = 0; i < fromMethods.length; i++) {
			fromMethod = fromMethods[i];
			fromMethodName = fromMethod.getName();
			if (!fromMethodName.contains("get"))
				continue;
			//      
			String str = fromMethodName.substring(3);
			if(!includesList.contains(str.substring(0,1).toLowerCase() + str.substring(1))) {
				continue;
			}
			toMethodName = "set" + fromMethodName.substring(3);
			toMethod = findMethodByName(toMethods, toMethodName);
			if (toMethod == null)
				continue;
			Object value = fromMethod.invoke(from, new Object[0]);
			if(value == null)
				continue;
			//       
			if(value instanceof Collection) {
				Collection newValue = (Collection)value;
				if(newValue.size() <= 0)
					continue;
			}
			toMethod.invoke(to, new Object[] {value});
		}
	}
	
	

	/**
	 *                
	 * 
	 * @param methods
	 * @param name
	 * @return
	 */
	public static Method findMethodByName(Method[] methods, String name) {
		for (int j = 0; j < methods.length; j++) {
			if (methods[j].getName().equals(name))
				return methods[j];
		}
		return null;
	}

2 apache commons-beanutilsのオープンソースを用いて実現する.
BeanUtils.copyProperties(dst, src).メソッドでは、ソースオブジェクトとターゲットオブジェクトの同じ名前のプロパティ値を過去にコピーできます.パラメータの前にターゲットオブジェクト、後ろにソースオブジェクトがあることに注意してください.このメソッドを使用するには、入口メソッドをソースオブジェクト、ターゲットオブジェクトのいずれかと同じソースファイルに配置することはできません.Noの場合、効果はありません.
PropertyUtils.copyProperties(dst, src).機能はBeanUtils.copyPropertiesと似ていますが、同じ属性名のタイプパラメータ間で変換操作を実行できます.
3 Springによる属性間のコピー.Springの内部には独自の実装方法があり、外でspringの管理レプリケーションを採用する必要がある場合は、springのソースコードを変更し、springのorg.springframework.beans.C achedIntrospectionResultsクラスのforClass、getPropertyDescriptor、getBeanInfoを可視に変更して再パッケージする必要があります.Springでコピーしたコードを抽出し、次のように修正します.
/**
	 *   spring  bean      
	 * @param source
	 * @param target
	 */
	@SuppressWarnings("unchecked")
	public static void copyPropertiesBySpring(Object source, Object target) throws Exception {
		Class actualEditable = target.getClass();
		PropertyDescriptor[] targetPds = getPropertyDescriptors(actualEditable);
		for (int i = 0; i < targetPds.length; i++) {
			PropertyDescriptor targetPd = targetPds[i];
			if (targetPd.getWriteMethod() != null) {
				PropertyDescriptor sourcePd = getPropertyDescriptor(source
						.getClass(), targetPd.getName());
				if (sourcePd != null && sourcePd.getReadMethod() != null) {
					try {
						Method readMethod = sourcePd.getReadMethod();
						if (!Modifier.isPublic(readMethod.getDeclaringClass()
								.getModifiers())) {
							readMethod.setAccessible(true);
						}
						Object value = readMethod.invoke(source, new Object[0]);
						if(value == null)
							continue;
						//       
						if(value instanceof Collection) {
//							Collection newValue = (Collection)value;
//							if(newValue.size() <= 0)
								continue;
						}
						Method writeMethod = targetPd.getWriteMethod();
						if (!Modifier.isPublic(writeMethod.getDeclaringClass()
								.getModifiers())) {
							writeMethod.setAccessible(true);
						}
						writeMethod.invoke(target, new Object[] { value });
					} catch (Throwable ex) {
					}
				}
			}
		}
	}	
	/**
	 *                
	 * @param clazz
	 * @param propertyName
	 * @return
	 * @throws BeansException
	 */
	@SuppressWarnings("unchecked")
	public static PropertyDescriptor getPropertyDescriptor(Class clazz,
			String propertyName) throws BeansException {
		CachedIntrospectionResults cr = CachedIntrospectionResults.forClass(clazz);
		return cr.getPropertyDescriptor(propertyName);
	}
	
	
	/**
	 *              
	 * @param clazz
	 * @return
	 * @throws BeansException
	 */
	@SuppressWarnings("unchecked")
	public static PropertyDescriptor[] getPropertyDescriptors(Class clazz) throws BeansException {
		CachedIntrospectionResults cr = CachedIntrospectionResults.forClass(clazz);
		return cr.getBeanInfo().getPropertyDescriptors();
	}