org.springframework.contributions.ioc.util.InheritanceSearch Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of spring-contributions Show documentation
Show all versions of spring-contributions Show documentation
This project adds a so called contribution mechanism (like known from Tapestry IOC or Eclipse Plugins) for configuration and extension of services to the Spring project.
// Licensed 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
//
// http://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.springframework.contributions.ioc.util;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.Set;
import org.springframework.contributions.ioc.services.ClassFabUtils;
/**
* Modified Copy of {@link org.apache.tapestry5.ioc.internal.util.InheritanceSearch} from Tapestry in version 5.2.5
*
* Used to search from a particular class up the inheritance hierarchy of extended classes and implemented interfaces.
*
* The search starts with the initial class (provided in the constructor). It progresses up the inheritance chain, but
* skips java.lang.Object.
*
* Once classes are exhausted, the inheritance hiearchy is searched. This is a breadth-first search, rooted in the
* interfaces implemented by the initial class at its super classes.
*
* Once all interfaces are exhausted, java.lang.Object is returned (it is always returned last).
*
* Two minor tweak to normal inheritance rules: - Normally, the parent class of an object array is
* java.lang.Object, which is odd because Foo[] is assignable to Object[]. Thus, we tweak the search so that the
* effective super class of Foo[] is Object[].
- The "super class" of a primtive type is its wrapper type,
* with the exception of void, whose "super class" is left at its normal value (Object.class)
*
* This class implements the {@link Iterable} interface, so it can be used directly in a for loop: for (Class
* search : new InheritanceSearch(startClass)) { ... }
*
* This class is not threadsafe.
*/
public class InheritanceSearch implements Iterator, Iterable
{
private Class searchClass;
private final Set addedInterfaces = CollectionFactory.newSet();
private final LinkedList interfaceQueue = CollectionFactory.newLinkedList();
private enum State
{
CLASS, INTERFACE, DONE
}
private State state;
public InheritanceSearch(Class searchClass)
{
this.searchClass = searchClass;
queueInterfaces(searchClass);
state = searchClass == Object.class ? State.INTERFACE : State.CLASS;
}
private void queueInterfaces(Class searchClass)
{
for (Class intf : searchClass.getInterfaces())
{
if (addedInterfaces.contains(intf)) continue;
interfaceQueue.addLast(intf);
addedInterfaces.add(intf);
}
}
public Iterator iterator()
{
return this;
}
public boolean hasNext()
{
return state != State.DONE;
}
public Class next()
{
switch (state)
{
case CLASS:
Class result = searchClass;
searchClass = parentOf(searchClass);
if (searchClass == null) state = State.INTERFACE;
else queueInterfaces(searchClass);
return result;
case INTERFACE:
if (interfaceQueue.isEmpty())
{
state = State.DONE;
return Object.class;
}
Class intf = interfaceQueue.removeFirst();
queueInterfaces(intf);
return intf;
default:
throw new IllegalStateException();
}
}
/**
* Returns the parent of the given class. Tweaks inheritance for object arrays. Returns null instead of
* Object.class.
*/
private Class parentOf(Class clazz)
{
if (clazz != void.class && clazz.isPrimitive()) return ClassFabUtils.getWrapperType(clazz);
if (clazz.isArray() && clazz != Object[].class)
{
Class componentType = clazz.getComponentType();
while (componentType.isArray()) componentType = componentType.getComponentType();
if (!componentType.isPrimitive()) return Object[].class;
}
Class parent = clazz.getSuperclass();
return parent != Object.class ? parent : null;
}
/**
* @throws UnsupportedOperationException always
*/
public void remove()
{
throw new UnsupportedOperationException();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy