
org.apache.commons.beanutils2.BeanMap Maven / Gradle / Ivy
Show all versions of commons-beanutils2 Show documentation
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* 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.apache.commons.beanutils2;
import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.AbstractMap;
import java.util.AbstractSet;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
import java.util.function.Function;
/**
* An implementation of Map for JavaBeans which uses introspection to get and put properties in the bean.
*
* If an exception occurs during attempts to get or set a property then the property is considered non existent in the Map
*
*/
public class BeanMap extends AbstractMap implements Cloneable {
/**
* Map entry used by {@link BeanMap}.
*/
protected static class Entry extends AbstractMap.SimpleEntry {
private static final long serialVersionUID = 1L;
/**
* The owner.
*/
private final BeanMap owner;
/**
* Constructs a new {@code Entry}.
*
* @param owner the BeanMap this entry belongs to
* @param key the key for this entry
* @param value the value for this entry
*/
protected Entry(final BeanMap owner, final String key, final Object value) {
super(key, value);
this.owner = owner;
}
/**
* Sets the value.
*
* @param value the new value for the entry
* @return the old value for the entry
*/
@Override
public Object setValue(final Object value) {
final String key = getKey();
final Object oldValue = owner.get(key);
owner.put(key, value);
final Object newValue = owner.get(key);
super.setValue(newValue);
return oldValue;
}
}
/**
* An empty array. Used to invoke accessors via reflection.
*/
public static final Object[] NULL_ARGUMENTS = {};
/**
* Maps primitive Class types to transformers. The transformer transform strings into the appropriate primitive wrapper.
*
* Private & unmodifiable replacement for the (public & static) defaultTransformers instance.
*/
private static final Map, Function, ?>> typeTransformers = Collections.unmodifiableMap(createTypeTransformers());
private static Map, Function, ?>> createTypeTransformers() {
final Map, Function, ?>> defTransformers = new HashMap<>();
defTransformers.put(Boolean.TYPE, input -> Boolean.valueOf(input.toString()));
defTransformers.put(Character.TYPE, input -> Character.valueOf(input.toString().charAt(0)));
defTransformers.put(Byte.TYPE, input -> Byte.valueOf(input.toString()));
defTransformers.put(Short.TYPE, input -> Short.valueOf(input.toString()));
defTransformers.put(Integer.TYPE, input -> Integer.valueOf(input.toString()));
defTransformers.put(Long.TYPE, input -> Long.valueOf(input.toString()));
defTransformers.put(Float.TYPE, input -> Float.valueOf(input.toString()));
defTransformers.put(Double.TYPE, input -> Double.valueOf(input.toString()));
return defTransformers;
}
private transient Object bean;
private final transient HashMap readMethods = new HashMap<>();
private final transient HashMap writeMethods = new HashMap<>();
private final transient HashMap> types = new HashMap<>();
/**
* Constructs a new empty {@code BeanMap}.
*/
public BeanMap() {
}
// Map interface
/**
* Constructs a new {@code BeanMap} that operates on the specified bean. If the given bean is {@code null}, then this map will be empty.
*
* @param bean the bean for this map to operate on
*/
public BeanMap(final Object bean) {
this.bean = bean;
initialize();
}
/**
* This method reinitializes the bean map to have default values for the bean's properties. This is accomplished by constructing a new instance of the bean
* which the map uses as its underlying data source. This behavior for {@code clear()} differs from the Map contract in that the mappings are not actually
* removed from the map (the mappings for a BeanMap are fixed).
*/
@Override
public void clear() {
if (bean == null) {
return;
}
Class extends Object> beanClass = null;
try {
beanClass = bean.getClass();
bean = beanClass.newInstance();
} catch (final Exception e) {
throw new UnsupportedOperationException("Could not create new instance of class: " + beanClass, e);
}
}
/**
* Clone this bean map using the following process:
*
*
* - If there is no underlying bean, return a cloned BeanMap without a bean.
*
- Since there is an underlying bean, try to instantiate a new bean of the same type using Class.newInstance().
*
- If the instantiation fails, throw a CloneNotSupportedException
*
- Clone the bean map and set the newly instantiated bean as the underlying bean for the bean map.
*
- Copy each property that is both readable and writable from the existing object to a cloned bean map.
*
- If anything fails along the way, throw a CloneNotSupportedException.
*
*
* @return a cloned instance of this bean map
* @throws CloneNotSupportedException if the underlying bean cannot be cloned
*/
@Override
public Object clone() throws CloneNotSupportedException {
final BeanMap newMap = (BeanMap) super.clone();
if (bean == null) {
// no bean, just an empty bean map at the moment. return a newly
// cloned and empty bean map.
return newMap;
}
Object newBean = null;
final Class extends Object> beanClass = bean.getClass(); // Cannot throw Exception
try {
newBean = beanClass.newInstance();
} catch (final Exception e) {
// unable to instantiate
final CloneNotSupportedException cnse = new CloneNotSupportedException(
"Unable to instantiate the underlying bean \"" + beanClass.getName() + "\": " + e);
cnse.initCause(e);
throw cnse;
}
try {
newMap.setBean(newBean);
} catch (final Exception e) {
final CloneNotSupportedException cnse = new CloneNotSupportedException("Unable to set bean in the cloned bean map: " + e);
cnse.initCause(e);
throw cnse;
}
try {
// copy only properties that are readable and writable. If its
// not readable, we can't get the value from the old map. If
// its not writable, we can't write a value into the new map.
readMethods.keySet().forEach(key -> {
if (getWriteMethod(key) != null) {
newMap.put(key, get(key));
}
});
} catch (final Exception e) {
final CloneNotSupportedException cnse = new CloneNotSupportedException("Unable to copy bean values to cloned bean map: " + e);
cnse.initCause(e);
throw cnse;
}
return newMap;
}
/**
* Returns true if the bean defines a property with the given name.
*
* The given name must be a {@code String}; if not, this method returns false. This method will also return false if the bean does not define a property
* with that name.
*
*
* Write-only properties will not be matched as the test operates against property read methods.
*
*
* @param name the name of the property to check
* @return false if the given name is null or is not a {@code String}; false if the bean does not define a property with that name; or true if the bean does
* define a property with that name
*/
@Override
public boolean containsKey(final Object name) {
return getReadMethod(name) != null;
}
/**
* Converts the given value to the given type. First, reflection is used to find a public constructor declared by the given class that takes one argument,
* which must be the precise type of the given value. If such a constructor is found, a new object is created by passing the given value to that
* constructor, and the newly constructed object is returned.
*
* If no such constructor exists, and the given type is a primitive type, then the given value is converted to a string using its {@link Object#toString()
* toString()} method, and that string is parsed into the correct primitive type using, for instance, {@link Integer#valueOf(String)} to convert the string
* into an {@code int}.
*
*
* If no special constructor exists and the given type is not a primitive type, this method returns the original value.
*
*
* @param The return type.
* @param newType the type to convert the value to
* @param value the value to convert
* @return the converted value
* @throws NumberFormatException if newType is a primitive type, and the string representation of the given value cannot be converted to that type
* @throws InstantiationException if the constructor found with reflection raises it
* @throws InvocationTargetException if the constructor found with reflection raises it
* @throws IllegalAccessException never
* @throws IllegalArgumentException never
*/
protected Object convertType(final Class newType, final Object value)
throws InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
// try call constructor
try {
final Constructor constructor = newType.getConstructor(value.getClass());
return constructor.newInstance(value);
} catch (final NoSuchMethodException e) {
// try using the transformers
final Function