2017-09-12 19:03:39 +02:00

224 lines
5.9 KiB
Java

/*
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code 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 General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package selectionresolution;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Set;
import java.util.Map;
/**
* A representation of a class/interface hierarchy graph (just the
* graph; the class data is represented elsewhere).
*/
public class HierarchyShape {
public static final int OBJECT_CLASS = -1;
protected int maxId;
/**
* The names of all the classes.
*/
private final HashSet<Integer> classes;
/**
* The names of all the interfaces.
*/
private final HashSet<Integer> interfaces;
private final HashMap<Integer, HashSet<Integer>> extensions;
/**
* Create an empty hierarchy shape.
*/
public HierarchyShape() {
this(0, new HashSet<>(), new HashSet<>(), new HashMap<>());
}
private HierarchyShape(final int maxId,
final HashSet<Integer> classes,
final HashSet<Integer> interfaces,
final HashMap<Integer, HashSet<Integer>> extensions) {
this.maxId = maxId;
this.classes = classes;
this.interfaces = interfaces;
this.extensions = extensions;
}
/**
* Make a copy of this hierarchy shape.
*/
public HierarchyShape copy() {
final HashMap<Integer, HashSet<Integer>> newextensions = new HashMap<>();
for(final Map.Entry<Integer, HashSet<Integer>> entry :
extensions.entrySet()) {
newextensions.put(entry.getKey(),
(HashSet<Integer>)entry.getValue().clone());
}
return new HierarchyShape(maxId, (HashSet<Integer>) classes.clone(),
(HashSet<Integer>) interfaces.clone(),
newextensions);
}
/**
* Add a class, and return its id.
*
* @return The new class id.
*/
public int addClass() {
final int id = maxId++;
classes.add(id);
return id;
}
/**
* Add an interface, and return its id.
*
* @return The new interface id.
*/
public int addInterface() {
final int id = maxId++;
interfaces.add(id);
return id;
}
/**
* Add an inheritance.
*
* @param sub The sub class/interface.
* @param sup The super class/interface
*/
public void addInherit(final int sub,
final int sup) {
HashSet<Integer> ext = extensions.get(sub);
if (ext == null) {
ext = new HashSet<>();
extensions.put(sub, ext);
}
ext.add(sup);
}
@Override
public String toString() {
String out = "";
for(int i = maxId - 1; i >= 0; i--) {
out += i + ": ";
for(int j = 0; j < maxId; j++) {
out += "[" + (inherits(i, j) ? "1" : "0") + "]";
}
out += "\n";
}
return out;
}
/**
* Indicate whether the first class inherits from the second.
*
* @param sub The possible subtype.
* @param sup The possible supertype.
* @return Whether or not {@code sub} inherits from {@code sup}.
*/
public boolean inherits(final int sub, final int sup) {
final Set<Integer> ext = extensions.get(sub);
if (ext != null) {
return ext.contains(sup);
} else {
return false;
}
}
/**
* Indicate whether a given type name is a class.
*
* @param id The type in question.
* @return Whether or not the type is a class.
*/
public boolean isClass(final int id) {
if (id == OBJECT_CLASS) {
return true;
}
return classes.contains(id);
}
/**
* Indicate whether a given type name is an interface.
*
* @param id The type in question.
* @return Whether or not the type is an interface.
*/
public boolean isInterface(final int id) {
if (id == OBJECT_CLASS) {
return false;
}
return interfaces.contains(id);
}
/**
* Get an iterator over the classes.
*
* @return An iterator over classes.
*/
public Collection<Integer> classes() {
return classes;
}
/**
* Get an iterator over the interfaces.
*
* @return An iterator over interfaces.
*/
public Collection<Integer> interfaces() {
return interfaces;
}
/**
* Get an iterator over all types.
*
* @return An iterator over all types.
*/
public Collection<Integer> types() {
final Set<Integer> combined = new HashSet(classes);
combined.addAll(interfaces);
return combined;
}
public int numClasses() {
return classes.size();
}
public int numInterfaces() {
return interfaces.size();
}
public int numTypes() {
return numClasses() + numInterfaces();
}
}