/* * This file is part of ELKI: * Environment for Developing KDD-Applications Supported by Index-Structures * * Copyright (C) 2017 * ELKI Development Team * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program 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 Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ package de.lmu.ifi.dbs.elki.data; import java.util.ArrayList; import java.util.Collections; import java.util.Iterator; import java.util.List; import de.lmu.ifi.dbs.elki.data.model.Model; import de.lmu.ifi.dbs.elki.data.type.SimpleTypeInformation; import de.lmu.ifi.dbs.elki.result.BasicResult; import de.lmu.ifi.dbs.elki.result.HierarchicalResult; import de.lmu.ifi.dbs.elki.result.Result; import de.lmu.ifi.dbs.elki.result.ResultUtil; import de.lmu.ifi.dbs.elki.utilities.datastructures.hierarchy.HashMapHierarchy; import de.lmu.ifi.dbs.elki.utilities.datastructures.hierarchy.Hierarchy; import de.lmu.ifi.dbs.elki.utilities.datastructures.hierarchy.ModifiableHierarchy; import de.lmu.ifi.dbs.elki.utilities.datastructures.iterator.It; /** * Result class for clusterings. Can be used for both hierarchical and * non-hierarchical clusterings. * * The class does not enforce or rely on clusterings to be a tree or DAG, * instead they can be an arbitrary forest of directed graphs that COULD contain * cycles. * * This class is NOT iterable for a simple reason: there is more than one method * to do so. You need to specify whether you want to use getToplevelClusters() * or getAllClusters(). * * @author Erich Schubert * @since 0.2 * * @apiviz.composedOf Cluster oneway - n * * @param <M> Model type */ public class Clustering<M extends Model> extends BasicResult { /** * Type information, for relation matching. */ public static final SimpleTypeInformation<Clustering<?>> TYPE = new SimpleTypeInformation<>(Clustering.class); /** * Keep a list of top level clusters. */ private List<Cluster<M>> toplevelclusters; /** * Cluster hierarchy. */ private ModifiableHierarchy<Cluster<M>> hierarchy; /** * Constructor with a list of top level clusters * * @param name The long name (for pretty printing) * @param shortname the short name (for filenames etc.) * @param toplevelclusters Top level clusters */ public Clustering(String name, String shortname, List<Cluster<M>> toplevelclusters) { super(name, shortname); this.toplevelclusters = toplevelclusters; this.hierarchy = new HashMapHierarchy<>(); for(Cluster<M> clus : toplevelclusters) { hierarchy.add(clus); } } /** * Constructor for an empty clustering * * @param name The long name (for pretty printing) * @param shortname the short name (for filenames etc.) */ public Clustering(String name, String shortname) { this(name, shortname, new ArrayList<Cluster<M>>()); } /** * Add a cluster to the clustering. * * @param clus new cluster */ public void addToplevelCluster(Cluster<M> clus) { toplevelclusters.add(clus); hierarchy.add(clus); } /** * Add a cluster to the clustering. * * @param parent Parent cluster * @param child Child cluster. */ public void addChildCluster(Cluster<M> parent, Cluster<M> child) { hierarchy.add(parent, child); } /** * Return top level clusters * * @return top level clusters */ public List<Cluster<M>> getToplevelClusters() { return toplevelclusters; } /** * Get the cluster hierarchy. * * @return Cluster hierarchy. */ public Hierarchy<Cluster<M>> getClusterHierarchy() { return hierarchy; } /** * Collect all clusters (recursively) into a List. * * @return List of all clusters. */ public List<Cluster<M>> getAllClusters() { ArrayList<Cluster<M>> res = new ArrayList<>(hierarchy.size()); for(It<Cluster<M>> iter = hierarchy.iterAll(); iter.valid(); iter.advance()) { res.add(iter.get()); } Collections.sort(res, Cluster.BY_NAME_SORTER); return res; } /** * Iterate over the top level clusters. * * @return Iterator */ public It<Cluster<M>> iterToplevelClusters() { return new It<Cluster<M>>() { Iterator<Cluster<M>> iter; Cluster<M> cur; { // Constructor. iter = toplevelclusters.iterator(); advance(); } @Override public boolean valid() { return cur != null; } @Override public It<Cluster<M>> advance() { if(iter.hasNext()) { cur = iter.next(); } else { cur = null; } return this; } @Override public Cluster<M> get() { return cur; } }; } /** * Collect all clustering results from a Result * * @param r Result * @return List of clustering results */ public static List<Clustering<? extends Model>> getClusteringResults(Result r) { if(r instanceof Clustering<?>) { List<Clustering<?>> crs = new ArrayList<>(1); crs.add((Clustering<?>) r); return crs; } if(r instanceof HierarchicalResult) { return ResultUtil.filterResults(((HierarchicalResult) r).getHierarchy(), r, Clustering.class); } return Collections.emptyList(); } }