/*
* 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
*
* 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.apache.bcel.generic;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.Map;
import org.apache.bcel.Constants;
import org.apache.bcel.Repository;
import org.apache.bcel.classfile.JavaClass;
/**
* Denotes reference such as java.lang.String.
*
* @version $Id: ObjectType.java 1600673 2014-06-05 14:19:44Z ggregory $
* @author <A HREF="mailto:m.dahm@gmx.de">M. Dahm</A>
*/
public class ObjectType extends ReferenceType {
private static final long serialVersionUID = -2819379966444533294L;
private final String class_name; // Class name of type
private static final int MAX_CACHE_ENTRIES = 200;
private static final int INITIAL_CACHE_CAPACITY = (int)(MAX_CACHE_ENTRIES/0.75);
private static HashMap<String, ObjectType> cache;
public synchronized static ObjectType getInstance(String class_name) {
if (cache == null) {
cache = new LinkedHashMap<String, ObjectType>(INITIAL_CACHE_CAPACITY, 0.75f, true) {
private static final long serialVersionUID = 2101159231109718724L;
@Override
protected boolean removeEldestEntry(Map.Entry<String, ObjectType> eldest) {
return size() > MAX_CACHE_ENTRIES;
}
};
}
ObjectType result = cache.get(class_name);
if (result != null) {
return result;
}
result = new ObjectType(class_name);
cache.put(class_name, result);
return result;
}
/**
* @param class_name fully qualified class name, e.g. java.lang.String
*/
public ObjectType(String class_name) {
super(Constants.T_REFERENCE, "L" + class_name.replace('.', '/') + ";");
this.class_name = class_name.replace('/', '.');
}
/** @return name of referenced class
*/
public String getClassName() {
return class_name;
}
/** @return a hash code value for the object.
*/
@Override
public int hashCode() {
return class_name.hashCode();
}
/** @return true if both type objects refer to the same class.
*/
@Override
public boolean equals( Object type ) {
return (type instanceof ObjectType)
? ((ObjectType) type).class_name.equals(class_name)
: false;
}
/**
* If "this" doesn't reference a class, it references an interface
* or a non-existant entity.
* @deprecated this method returns an inaccurate result
* if the class or interface referenced cannot
* be found: use referencesClassExact() instead
*/
@Deprecated
public boolean referencesClass() {
try {
JavaClass jc = Repository.lookupClass(class_name);
return jc.isClass();
} catch (ClassNotFoundException e) {
return false;
}
}
/**
* If "this" doesn't reference an interface, it references a class
* or a non-existant entity.
* @deprecated this method returns an inaccurate result
* if the class or interface referenced cannot
* be found: use referencesInterfaceExact() instead
*/
@Deprecated
public boolean referencesInterface() {
try {
JavaClass jc = Repository.lookupClass(class_name);
return !jc.isClass();
} catch (ClassNotFoundException e) {
return false;
}
}
/**
* Return true if this type references a class,
* false if it references an interface.
* @return true if the type references a class, false if
* it references an interface
* @throws ClassNotFoundException if the class or interface
* referenced by this type can't be found
*/
public boolean referencesClassExact() throws ClassNotFoundException {
JavaClass jc = Repository.lookupClass(class_name);
return jc.isClass();
}
/**
* Return true if this type references an interface,
* false if it references a class.
* @return true if the type references an interface, false if
* it references a class
* @throws ClassNotFoundException if the class or interface
* referenced by this type can't be found
*/
public boolean referencesInterfaceExact() throws ClassNotFoundException {
JavaClass jc = Repository.lookupClass(class_name);
return !jc.isClass();
}
/**
* Return true if this type is a subclass of given ObjectType.
* @throws ClassNotFoundException if any of this class's superclasses
* can't be found
*/
public boolean subclassOf( ObjectType superclass ) throws ClassNotFoundException {
if (this.referencesInterface() || superclass.referencesInterface()) {
return false;
}
return Repository.instanceOf(this.class_name, superclass.class_name);
}
/**
* Java Virtual Machine Specification edition 2, � 5.4.4 Access Control
* @throws ClassNotFoundException if the class referenced by this type
* can't be found
*/
public boolean accessibleTo( ObjectType accessor ) throws ClassNotFoundException {
JavaClass jc = Repository.lookupClass(class_name);
if (jc.isPublic()) {
return true;
} else {
JavaClass acc = Repository.lookupClass(accessor.class_name);
return acc.getPackageName().equals(jc.getPackageName());
}
}
}