org.jboss.marshalling.reflect.ReflectiveCreator Maven / Gradle / Ivy
/*
* JBoss, Home of Professional Open Source
* Copyright 2008, JBoss Inc., and individual contributors as indicated
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.jboss.marshalling.reflect;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.util.concurrent.ConcurrentMap;
import java.util.EnumSet;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.security.Permission;
import java.io.InvalidClassException;
import java.io.SerializablePermission;
import org.jboss.marshalling.Creator;
/**
* A creator that simply uses reflection to locate and invoke a zero-argument constructor.
*/
public class ReflectiveCreator implements Creator {
/**
* Constructor cache. Keys and values are weak, and keys and values both use identity comparisons.
*/
private static final ConcurrentMap, Constructor>> constructorMap = new ConcurrentReferenceHashMap, Constructor>>(1000, 0.5f, 16, ConcurrentReferenceHashMap.ReferenceType.WEAK, ConcurrentReferenceHashMap.ReferenceType.WEAK, EnumSet.of(ConcurrentReferenceHashMap.Option.IDENTITY_COMPARISONS));
private static final Permission CREATOR_PERM = new SerializablePermission("creator");
public ReflectiveCreator() {
final SecurityManager sm = System.getSecurityManager();
if (sm != null) {
sm.checkPermission(CREATOR_PERM);
}
}
/**
* Get the constructor to use for a class. Returns {@code null} if no suitable constructor is available.
*
* @param clazz the class to get a constructor for
* @return the constructor, or {@code null} if none is available
*/
protected Constructor getNewConstructor(final Class clazz) {
final Constructor newConstructor = AccessController.doPrivileged(new PrivilegedAction>() {
public Constructor run() {
try {
final Constructor constructor = clazz.getDeclaredConstructor();
constructor.setAccessible(true);
return constructor;
} catch (NoSuchMethodException e) {
return null;
}
}
});
return newConstructor;
}
/**
* Get a cached constructor for the class. If no such constructor is found in the cache, the
* {@code getNewConstructor()} method is invoked to get it. If no constructor is available then an
* {@code InstantiationException} is thrown.
*
* @param clazz the class to look up
* @return the cached constructor
* @throws InstantiationException if no suitable constructor is available
*/
@SuppressWarnings({"unchecked"})
protected Constructor getCachedConstructor(final Class clazz) throws InvalidClassException {
final Constructor> constructor = constructorMap.get(clazz);
if (constructor != null) {
return (Constructor) constructor;
}
final Constructor newConstructor = getNewConstructor(clazz);
if (newConstructor == null) {
throw new InvalidClassException(clazz.getName(), "No suitable constructor could be found");
}
final Constructor> old = constructorMap.putIfAbsent(clazz, newConstructor);
if (old != null) {
return (Constructor) old;
} else {
return newConstructor;
}
}
/** {@inheritDoc} */
public T create(final Class clazz) throws InvalidClassException {
final Constructor constructor = getCachedConstructor(clazz);
try {
return constructor.newInstance();
} catch (InvocationTargetException e) {
final InvalidClassException ice = new InvalidClassException(clazz.getName(), "Constructor threw an exception");
ice.initCause(e);
throw ice;
} catch (IllegalAccessException e) {
throw new InvalidClassException(clazz.getName(), "Illegal access exception occurred accessing the constructor: " + String.valueOf(e));
} catch (InstantiationException e) {
throw new InvalidClassException(clazz.getName(), "Instantiation exception: " + String.valueOf(e));
}
}
}