Title: | Fast and Robust Hierarchical Clustering with Noise Points Detection |
---|---|
Description: | A retake on the Genie algorithm (Gagolewski, 2021 <DOI:10.1016/j.softx.2021.100722>) - a robust hierarchical clustering method (Gagolewski, Bartoszuk, Cena, 2016 <DOI:10.1016/j.ins.2016.05.003>). Now faster and more memory efficient; determining the whole hierarchy for datasets of 10M points in low dimensional Euclidean spaces or 100K points in high-dimensional ones takes only 1-2 minutes. Allows clustering with respect to mutual reachability distances so that it can act as a noise point detector or a robustified version of 'HDBSCAN*' (that is able to detect a predefined number of clusters and hence it does not dependent on the somewhat fragile 'eps' parameter). The package also features an implementation of inequality indices (the Gini, Bonferroni index), external cluster validity measures (e.g., the normalised clustering accuracy and partition similarity scores such as the adjusted Rand, Fowlkes-Mallows, adjusted mutual information, and the pair sets index), and internal cluster validity indices (e.g., the Calinski-Harabasz, Davies-Bouldin, Ball-Hall, Silhouette, and generalised Dunn indices). See also the 'Python' version of 'genieclust' available on 'PyPI', which supports sparse data, more metrics, and even larger datasets. |
Authors: | Marek Gagolewski [aut, cre, cph] , Maciej Bartoszuk [ctb], Anna Cena [ctb], Peter M. Larsen [ctb] |
Maintainer: | Marek Gagolewski <[email protected]> |
License: | AGPL-3 |
Version: | 1.1.6 |
Built: | 2025-01-19 06:01:45 UTC |
Source: | https://github.com/gagolews/genieclust |
Implementation of a number of so-called cluster validity indices critically reviewed in (Gagolewski, Bartoszuk, Cena, 2021). See Section 2 therein and (Gagolewski, 2022) for the respective definitions.
The greater the index value, the more valid (whatever that means) the assessed partition. For consistency, the Ball-Hall and Davies-Bouldin indexes as well as the within-cluster sum of squares (WCSS) take negative values.
calinski_harabasz_index(X, y) dunnowa_index( X, y, M = 25L, owa_numerator = "SMin:5", owa_denominator = "Const" ) generalised_dunn_index(X, y, lowercase_d, uppercase_d) negated_ball_hall_index(X, y) negated_davies_bouldin_index(X, y) negated_wcss_index(X, y) silhouette_index(X, y) silhouette_w_index(X, y) wcnn_index(X, y, M = 25L)
calinski_harabasz_index(X, y) dunnowa_index( X, y, M = 25L, owa_numerator = "SMin:5", owa_denominator = "Const" ) generalised_dunn_index(X, y, lowercase_d, uppercase_d) negated_ball_hall_index(X, y) negated_davies_bouldin_index(X, y) negated_wcss_index(X, y) silhouette_index(X, y) silhouette_w_index(X, y) wcnn_index(X, y, M = 25L)
X |
numeric matrix with |
y |
vector of |
M |
number of nearest neighbours |
owa_numerator , owa_denominator
|
single string specifying
the OWA operators to use in the definition of the DuNN index;
one of: |
lowercase_d |
an integer between 1 and 5, denoting
|
uppercase_d |
an integer between 1 and 3, denoting
|
A single numeric value (the more, the better).
Marek Gagolewski and other contributors
Ball G.H., Hall D.J., ISODATA: A novel method of data analysis and pattern classification, Technical report No. AD699616, Stanford Research Institute, 1965.
Bezdek J., Pal N., Some new indexes of cluster validity, IEEE Transactions on Systems, Man, and Cybernetics, Part B 28, 1998, 301-315, doi:10.1109/3477.678624.
Calinski T., Harabasz J., A dendrite method for cluster analysis, Communications in Statistics 3(1), 1974, 1-27, doi:10.1080/03610927408827101.
Davies D.L., Bouldin D.W., A Cluster Separation Measure, IEEE Transactions on Pattern Analysis and Machine Intelligence PAMI-1 (2), 1979, 224-227, doi:10.1109/TPAMI.1979.4766909.
Dunn J.C., A Fuzzy Relative of the ISODATA Process and Its Use in Detecting Compact Well-Separated Clusters, Journal of Cybernetics 3(3), 1973, 32-57, doi:10.1080/01969727308546046.
Gagolewski M., Bartoszuk M., Cena A., Are cluster validity measures (in)valid?, Information Sciences 581, 620-636, 2021, doi:10.1016/j.ins.2021.10.004; preprint: https://raw.githubusercontent.com/gagolews/bibliography/master/preprints/2021cvi.pdf.
Gagolewski M., A Framework for Benchmarking Clustering Algorithms, SoftwareX 20, 2022, 101270, doi:10.1016/j.softx.2022.101270, https://clustering-benchmarks.gagolewski.com.
Rousseeuw P.J., Silhouettes: A Graphical Aid to the Interpretation and Validation of Cluster Analysis, Computational and Applied Mathematics 20, 1987, 53-65, doi:10.1016/0377-0427(87)90125-7.
The official online manual of genieclust at https://genieclust.gagolewski.com/
Gagolewski M., genieclust: Fast and robust hierarchical clustering, SoftwareX 15:100722, 2021, doi:10.1016/j.softx.2021.100722.
X <- as.matrix(iris[,1:4]) X[,] <- jitter(X) # otherwise we get a non-unique solution y <- as.integer(iris[[5]]) calinski_harabasz_index(X, y) # good calinski_harabasz_index(X, sample(1:3, nrow(X), replace=TRUE)) # bad
X <- as.matrix(iris[,1:4]) X[,] <- jitter(X) # otherwise we get a non-unique solution y <- as.integer(iris[[5]]) calinski_harabasz_index(X, y) # good calinski_harabasz_index(X, sample(1:3, nrow(X), replace=TRUE)) # bad
The functions described in this section quantify the similarity between
two label vectors x
and y
which represent two partitions
of a set of elements into, respectively,
and
nonempty and pairwise disjoint subsets.
For instance, x
and y
can represent two clusterings
of a dataset with observations specified by two vectors
of labels. The functions described here can be used as external cluster
validity measures, where we assume that
x
is
a reference (ground-truth) partition whilst y
is the vector
of predicted cluster memberships.
All indices except normalized_clustering_accuracy()
can act as a pairwise partition similarity score: they are symmetric,
i.e., index(x, y) == index(y, x)
.
Each index except mi_score()
(which computes the mutual
information score) outputs 1 given two identical partitions.
Note that partitions are always defined up to a permutation (bijection)
of the set of possible labels, e.g., (1, 1, 2, 1) and (4, 4, 2, 4)
represent the same 2-partition.
normalized_clustering_accuracy(x, y = NULL) normalized_pivoted_accuracy(x, y = NULL) pair_sets_index(x, y = NULL, simplified = FALSE, clipped = TRUE) adjusted_rand_score(x, y = NULL, clipped = FALSE) rand_score(x, y = NULL) adjusted_fm_score(x, y = NULL, clipped = FALSE) fm_score(x, y = NULL) mi_score(x, y = NULL) normalized_mi_score(x, y = NULL) adjusted_mi_score(x, y = NULL, clipped = FALSE) normalized_confusion_matrix(x, y = NULL) normalizing_permutation(x, y = NULL)
normalized_clustering_accuracy(x, y = NULL) normalized_pivoted_accuracy(x, y = NULL) pair_sets_index(x, y = NULL, simplified = FALSE, clipped = TRUE) adjusted_rand_score(x, y = NULL, clipped = FALSE) rand_score(x, y = NULL) adjusted_fm_score(x, y = NULL, clipped = FALSE) fm_score(x, y = NULL) mi_score(x, y = NULL) normalized_mi_score(x, y = NULL) adjusted_mi_score(x, y = NULL, clipped = FALSE) normalized_confusion_matrix(x, y = NULL) normalizing_permutation(x, y = NULL)
x |
an integer vector of length n (or an object coercible to)
representing a K-partition of an n-set (e.g., a reference partition),
or a confusion matrix with K rows and L columns
(see |
y |
an integer vector of length n (or an object coercible to)
representing an L-partition of the same set (e.g., the output of a
clustering algorithm we wish to compare with |
simplified |
whether to assume E=1 in the definition of the pair sets index index, i.e., use Eq. (20) in (Rezaei, Franti, 2016) instead of Eq. (18) |
clipped |
whether the result should be clipped to the unit interval, i.e., [0, 1] |
normalized_clustering_accuracy()
(Gagolewski, 2023)
is an asymmetric external cluster validity measure
which assumes that the label vector x
(or rows in the confusion
matrix) represents the reference (ground truth) partition.
It is an average proportion of correctly classified points in each cluster
above the worst case scenario of uniform membership assignment,
with cluster ID matching based on the solution to the maximal linear
sum assignment problem; see normalized_confusion_matrix
).
It is given by:
,
where
is a confusion matrix with
rows and
columns,
is a permutation of the set
, and
is the i-th row sum,
under the assumption that
for
or
and
.
normalized_pivoted_accuracy()
is defined as
,
where
is a permutation of the set
,
and
is the sum of all elements in
.
For non-square matrices, missing rows/columns are assumed
to be filled with 0s.
pair_sets_index()
(PSI) was introduced in (Rezaei, Franti, 2016).
The simplified PSI assumes E=1 in the definition of the index,
i.e., uses Eq. (20) in the said paper instead of Eq. (18).
For non-square matrices, missing rows/columns are assumed
to be filled with 0s.
rand_score()
gives the Rand score (the "probability" of agreement
between the two partitions) and
adjusted_rand_score()
is its version corrected for chance,
see (Hubert, Arabie, 1985): its expected value is 0 given two independent
partitions. Due to the adjustment, the resulting index may be negative
for some inputs.
Similarly, fm_score()
gives the Fowlkes-Mallows (FM) score
and adjusted_fm_score()
is its adjusted-for-chance version;
see (Hubert, Arabie, 1985).
mi_score()
, adjusted_mi_score()
and
normalized_mi_score()
are information-theoretic
scores, based on mutual information,
see the definition of and
in (Vinh et al., 2010).
normalized_confusion_matrix()
computes the confusion matrix
and permutes its rows and columns so that the sum of the elements
of the main diagonal is the largest possible (by solving
the maximal assignment problem).
The function only accepts .
The reordering of the columns of a confusion matrix can be determined
by calling
normalizing_permutation()
.
Also note that the built-in
table()
determines the standard confusion matrix.
Each cluster validity measure is a single numeric value.
normalized_confusion_matrix()
returns a numeric matrix.
normalizing_permutation()
returns a vector of indexes.
Marek Gagolewski and other contributors
Gagolewski M., A framework for benchmarking clustering algorithms, SoftwareX 20, 2022, 101270, doi:10.1016/j.softx.2022.101270, https://clustering-benchmarks.gagolewski.com.
Gagolewski M., Normalised clustering accuracy: An asymmetric external cluster validity measure, Journal of Classification, 2024, in press, doi:10.1007/s00357-024-09482-2.
Hubert L., Arabie P., Comparing partitions, Journal of Classification 2(1), 1985, 193-218, esp. Eqs. (2) and (4).
Meila M., Heckerman D., An experimental comparison of model-based clustering methods, Machine Learning 42, 2001, pp. 9-29, doi:10.1023/A:1007648401407.
Rezaei M., Franti P., Set matching measures for external cluster validity, IEEE Transactions on Knowledge and Data Mining 28(8), 2016, 2173-2186.
Steinley D., Properties of the Hubert-Arabie adjusted Rand index, Psychological Methods 9(3), 2004, pp. 386-396, doi:10.1037/1082-989X.9.3.386.
Vinh N.X., Epps J., Bailey J., Information theoretic measures for clusterings comparison: Variants, properties, normalization and correction for chance, Journal of Machine Learning Research 11, 2010, 2837-2854.
The official online manual of genieclust at https://genieclust.gagolewski.com/
Gagolewski M., genieclust: Fast and robust hierarchical clustering, SoftwareX 15:100722, 2021, doi:10.1016/j.softx.2021.100722.
y_true <- iris[[5]] y_pred <- kmeans(as.matrix(iris[1:4]), 3)$cluster normalized_clustering_accuracy(y_true, y_pred) normalized_pivoted_accuracy(y_true, y_pred) pair_sets_index(y_true, y_pred) pair_sets_index(y_true, y_pred, simplified=TRUE) adjusted_rand_score(y_true, y_pred) rand_score(table(y_true, y_pred)) # the same adjusted_fm_score(y_true, y_pred) fm_score(y_true, y_pred) mi_score(y_true, y_pred) normalized_mi_score(y_true, y_pred) adjusted_mi_score(y_true, y_pred) normalized_confusion_matrix(y_true, y_pred) normalizing_permutation(y_true, y_pred)
y_true <- iris[[5]] y_pred <- kmeans(as.matrix(iris[1:4]), 3)$cluster normalized_clustering_accuracy(y_true, y_pred) normalized_pivoted_accuracy(y_true, y_pred) pair_sets_index(y_true, y_pred) pair_sets_index(y_true, y_pred, simplified=TRUE) adjusted_rand_score(y_true, y_pred) rand_score(table(y_true, y_pred)) # the same adjusted_fm_score(y_true, y_pred) fm_score(y_true, y_pred) mi_score(y_true, y_pred) normalized_mi_score(y_true, y_pred) adjusted_mi_score(y_true, y_pred) normalized_confusion_matrix(y_true, y_pred) normalizing_permutation(y_true, y_pred)
Provides access to the implementation of the Dual-Tree Boruvka
algorithm from the mlpack
package (if available).
It is based on kd-trees and is fast for (very) low-dimensional
Euclidean spaces. For higher dimensional spaces (say, over 5 features)
or other metrics, use the parallelised Prim-like algorithm implemented
in mst()
.
emst_mlpack(X, leaf_size = 1, naive = FALSE, verbose = FALSE)
emst_mlpack(X, leaf_size = 1, naive = FALSE, verbose = FALSE)
X |
a numeric matrix (or an object coercible to one, e.g., a data frame with numeric-like columns) |
leaf_size |
size of leaves in the kd-tree, controls the trade-off between speed and memory consumption |
naive |
logical; whether to use the naive, quadratic-time algorithm |
verbose |
logical; whether to print diagnostic messages |
An object of class mst
, see mst()
for details.
Marek Gagolewski and other contributors
March W.B., Ram P., Gray A.G., Fast Euclidean Minimum Spanning Tree: Algorithm, Analysis, and Applications, Proc. ACM SIGKDD'10, 2010, 603-611, https://mlpack.org/papers/emst.pdf.
Curtin R.R., Edel M., Lozhnikov M., Mentekidis Y., Ghaisas S., Zhang S., mlpack 3: A fast, flexible machine learning library, Journal of Open Source Software 3(26), 2018, 726.
The official online manual of genieclust at https://genieclust.gagolewski.com/
Gagolewski M., genieclust: Fast and robust hierarchical clustering, SoftwareX 15:100722, 2021, doi:10.1016/j.softx.2021.100722.
A reimplementation of Genie - a robust and outlier resistant
clustering algorithm (see Gagolewski, Bartoszuk, Cena, 2016).
The Genie algorithm is based on a minimum spanning tree (MST) of the
pairwise distance graph of a given point set.
Just like the single linkage, it consumes the edges
of the MST in an increasing order of weights. However, it prevents
the formation of clusters of highly imbalanced sizes; once the Gini index
(see gini_index()
) of the cluster size distribution
raises above gini_threshold
, a forced merge of a point group
of the smallest size is performed. Its appealing simplicity goes hand
in hand with its usability; Genie often outperforms
other clustering approaches on benchmark data,
such as https://github.com/gagolews/clustering-benchmarks.
The clustering can now also be computed with respect to the
mutual reachability distance (based, e.g., on the Euclidean metric),
which is used in the definition of the HDBSCAN* algorithm
(see Campello et al., 2013). If M
> 1, then the mutual reachability
distance with smoothing factor
M
is used instead of the
chosen "raw" distance . It holds
,
where
is
with
being the
(
M
-1)-th nearest neighbour of .
This makes "noise" and "boundary" points being "pulled away" from each other.
The Genie correction together with the smoothing factor M
> 1 (note that
M
= 2 corresponds to the original distance) gives a robustified version of
the HDBSCAN* algorithm that is able to detect a predefined number of
clusters. Hence it does not dependent on the DBSCAN's somewhat magical
eps
parameter or the HDBSCAN's min_cluster_size
one.
gclust(d, ...) ## Default S3 method: gclust( d, gini_threshold = 0.3, distance = c("euclidean", "l2", "manhattan", "cityblock", "l1", "cosine"), cast_float32 = TRUE, verbose = FALSE, ... ) ## S3 method for class 'dist' gclust(d, gini_threshold = 0.3, verbose = FALSE, ...) ## S3 method for class 'mst' gclust(d, gini_threshold = 0.3, verbose = FALSE, ...) genie(d, ...) ## Default S3 method: genie( d, k, gini_threshold = 0.3, distance = c("euclidean", "l2", "manhattan", "cityblock", "l1", "cosine"), M = 1L, postprocess = c("boundary", "none", "all"), detect_noise = M > 1L, cast_float32 = TRUE, verbose = FALSE, ... ) ## S3 method for class 'dist' genie( d, k, gini_threshold = 0.3, M = 1L, postprocess = c("boundary", "none", "all"), detect_noise = M > 1L, verbose = FALSE, ... ) ## S3 method for class 'mst' genie( d, k, gini_threshold = 0.3, postprocess = c("boundary", "none", "all"), detect_noise = FALSE, verbose = FALSE, ... )
gclust(d, ...) ## Default S3 method: gclust( d, gini_threshold = 0.3, distance = c("euclidean", "l2", "manhattan", "cityblock", "l1", "cosine"), cast_float32 = TRUE, verbose = FALSE, ... ) ## S3 method for class 'dist' gclust(d, gini_threshold = 0.3, verbose = FALSE, ...) ## S3 method for class 'mst' gclust(d, gini_threshold = 0.3, verbose = FALSE, ...) genie(d, ...) ## Default S3 method: genie( d, k, gini_threshold = 0.3, distance = c("euclidean", "l2", "manhattan", "cityblock", "l1", "cosine"), M = 1L, postprocess = c("boundary", "none", "all"), detect_noise = M > 1L, cast_float32 = TRUE, verbose = FALSE, ... ) ## S3 method for class 'dist' genie( d, k, gini_threshold = 0.3, M = 1L, postprocess = c("boundary", "none", "all"), detect_noise = M > 1L, verbose = FALSE, ... ) ## S3 method for class 'mst' genie( d, k, gini_threshold = 0.3, postprocess = c("boundary", "none", "all"), detect_noise = FALSE, verbose = FALSE, ... )
d |
a numeric matrix (or an object coercible to one,
e.g., a data frame with numeric-like columns) or an
object of class |
... |
further arguments passed to other methods. |
gini_threshold |
threshold for the Genie correction, i.e., the Gini index of the cluster size distribution; Threshold of 1.0 disables the correction. Low thresholds highly penalise the formation of small clusters. |
distance |
metric used to compute the linkage, one of:
|
cast_float32 |
logical; whether to compute the distances using 32-bit instead of 64-bit precision floating-point arithmetic (up to 2x faster). |
verbose |
logical; whether to print diagnostic messages and progress information. |
k |
the desired number of clusters to detect, |
M |
smoothing factor; |
postprocess |
one of |
detect_noise |
whether the minimum spanning tree's leaves
should be marked as noise points, defaults to |
Note that, as in the case of all the distance-based methods, the standardisation of the input features is definitely worth giving a try.
If d
is a numeric matrix or an object of class dist
,
mst()
will be called to compute an MST, which generally
takes at most time (the algorithm we provide is parallelised,
environment variable
OMP_NUM_THREADS
controls the number of threads
in use). However, see emst_mlpack()
for a very fast alternative
in the case of Euclidean spaces of (very) low dimensionality and M
= 1.
Given an minimum spanning tree, the algorithm runs in time.
Therefore, if you want to test different
gini_threshold
s,
(or k
s), it is best to explicitly compute the MST first.
According to the algorithm's original definition,
the resulting partition tree (dendrogram) might violate
the ultrametricity property (merges might occur at levels that
are not increasing w.r.t. a between-cluster distance).
gclust()
automatically corrects departures from
ultrametricity by applying height = rev(cummin(rev(height)))
.
gclust()
computes the whole clustering hierarchy; it
returns a list of class hclust
,
see hclust
. Use cutree
to obtain
an arbitrary k-partition.
genie()
returns a k
-partition - a vector with elements in 1,...,k,
whose i-th element denotes the i-th input point's cluster identifier.
Missing values (NA
) denote noise points (if detect_noise
is TRUE
).
Marek Gagolewski and other contributors
Gagolewski M., Bartoszuk M., Cena A., Genie: A new, fast, and outlier-resistant hierarchical clustering algorithm, Information Sciences 363, 2016, 8-23, doi:10.1016/j.ins.2016.05.003.
Campello R.J.G.B., Moulavi D., Sander J., Density-based clustering based on hierarchical density estimates, Lecture Notes in Computer Science 7819, 2013, 160-172, doi:10.1007/978-3-642-37456-2_14.
Gagolewski M., Cena A., Bartoszuk M., Brzozowski L., Clustering with minimum spanning trees: How good can it be?, Journal of Classification, 2024, in press, doi:10.1007/s00357-024-09483-1.
The official online manual of genieclust at https://genieclust.gagolewski.com/
Gagolewski M., genieclust: Fast and robust hierarchical clustering, SoftwareX 15:100722, 2021, doi:10.1016/j.softx.2021.100722.
mst()
for the minimum spanning tree routines.
adjusted_rand_score()
(amongst others) for external
cluster validity measures (partition similarity scores).
library("datasets") data("iris") X <- iris[1:4] h <- gclust(X) y_pred <- cutree(h, 3) y_test <- iris[,5] plot(iris[,2], iris[,3], col=y_pred, pch=as.integer(iris[,5]), asp=1, las=1) adjusted_rand_score(y_test, y_pred) pair_sets_index(y_test, y_pred) # Fast for low-dimensional Euclidean spaces: # h <- gclust(emst_mlpack(X))
library("datasets") data("iris") X <- iris[1:4] h <- gclust(X) y_pred <- cutree(h, 3) y_test <- iris[,5] plot(iris[,2], iris[,3], col=y_pred, pch=as.integer(iris[,5]), asp=1, las=1) adjusted_rand_score(y_test, y_pred) pair_sets_index(y_test, y_pred) # Fast for low-dimensional Euclidean spaces: # h <- gclust(emst_mlpack(X))
gini_index()
gives the normalised Gini index,
bonferroni_index()
implements the Bonferroni index, and
devergottini_index()
implements the De Vergottini index.
gini_index(x) bonferroni_index(x) devergottini_index(x)
gini_index(x) bonferroni_index(x) devergottini_index(x)
x |
numeric vector of non-negative values |
These indices can be used to quantify the "inequality" of a numeric sample.
They can be conceived as normalised measures of data dispersion.
For constant vectors (perfect equity), the indices yield values of 0.
Vectors with all elements but one equal to 0 (perfect inequality),
are assigned scores of 1.
They follow the Pigou-Dalton principle (are Schur-convex):
setting and
with
and
(taking from the "rich" and
giving to the "poor") decreases the inequality
These indices have applications in economics, amongst others. The Genie clustering algorithm uses the Gini index as a measure of the inequality of cluster sizes.
The normalised Gini index is given by:
The normalised Bonferroni index is given by:
The normalised De Vergottini index is given by:
Here, is an ordering permutation of
.
Time complexity: for sorted (increasingly) data.
Otherwise, the vector will be sorted.
The value of the inequality index, a number in .
Marek Gagolewski and other contributors
Bonferroni C., Elementi di Statistica Generale, Libreria Seber, Firenze, 1930.
Gagolewski M., Bartoszuk M., Cena A., Genie: A new, fast, and outlier-resistant hierarchical clustering algorithm, Information Sciences 363, 2016, pp. 8-23. doi:10.1016/j.ins.2016.05.003
Gini C., Variabilita e Mutabilita, Tipografia di Paolo Cuppini, Bologna, 1912.
The official online manual of genieclust at https://genieclust.gagolewski.com/
Gagolewski M., genieclust: Fast and robust hierarchical clustering, SoftwareX 15:100722, 2021, doi:10.1016/j.softx.2021.100722.
gini_index(c(2, 2, 2, 2, 2)) # no inequality gini_index(c(0, 0, 10, 0, 0)) # one has it all gini_index(c(7, 0, 3, 0, 0)) # give to the poor, take away from the rich gini_index(c(6, 0, 3, 1, 0)) # (a.k.a. Pigou-Dalton principle) bonferroni_index(c(2, 2, 2, 2, 2)) bonferroni_index(c(0, 0, 10, 0, 0)) bonferroni_index(c(7, 0, 3, 0, 0)) bonferroni_index(c(6, 0, 3, 1, 0)) devergottini_index(c(2, 2, 2, 2, 2)) devergottini_index(c(0, 0, 10, 0, 0)) devergottini_index(c(7, 0, 3, 0, 0)) devergottini_index(c(6, 0, 3, 1, 0))
gini_index(c(2, 2, 2, 2, 2)) # no inequality gini_index(c(0, 0, 10, 0, 0)) # one has it all gini_index(c(7, 0, 3, 0, 0)) # give to the poor, take away from the rich gini_index(c(6, 0, 3, 1, 0)) # (a.k.a. Pigou-Dalton principle) bonferroni_index(c(2, 2, 2, 2, 2)) bonferroni_index(c(0, 0, 10, 0, 0)) bonferroni_index(c(7, 0, 3, 0, 0)) bonferroni_index(c(6, 0, 3, 1, 0)) devergottini_index(c(2, 2, 2, 2, 2)) devergottini_index(c(0, 0, 10, 0, 0)) devergottini_index(c(7, 0, 3, 0, 0)) devergottini_index(c(6, 0, 3, 1, 0))
An parallelised implementation of a Jarnik (Prim/Dijkstra)-like algorithm for determining a(*) minimum spanning tree (MST) of a complete undirected graph representing a set of n points with weights given by a pairwise distance matrix.
(*) Note that there might be multiple minimum trees spanning a given graph.
mst(d, ...) ## Default S3 method: mst( d, distance = c("euclidean", "l2", "manhattan", "cityblock", "l1", "cosine"), M = 1L, cast_float32 = TRUE, verbose = FALSE, ... ) ## S3 method for class 'dist' mst(d, M = 1L, verbose = FALSE, ...)
mst(d, ...) ## Default S3 method: mst( d, distance = c("euclidean", "l2", "manhattan", "cityblock", "l1", "cosine"), M = 1L, cast_float32 = TRUE, verbose = FALSE, ... ) ## S3 method for class 'dist' mst(d, M = 1L, verbose = FALSE, ...)
d |
either a numeric matrix (or an object coercible to one,
e.g., a data frame with numeric-like columns) or an
object of class |
... |
further arguments passed to or from other methods |
distance |
metric used to compute the linkage, one of:
|
M |
smoothing factor; |
cast_float32 |
logical; whether to compute the distances using 32-bit instead of 64-bit precision floating-point arithmetic (up to 2x faster) |
verbose |
logical; whether to print diagnostic messages and progress information |
If d
is a numeric matrix of size ,
the
distances are computed on the fly, so that
memory is used.
The algorithm is parallelised; set the OMP_NUM_THREADS
environment
variable Sys.setenv
to control the number of threads
used.
Time complexity is for the method accepting an object of
class
dist
and otherwise.
If M
>= 2, then the mutual reachability distance
with smoothing factor
M
(see Campello et al. 2013)
is used instead of the chosen "raw" distance .
It holds
, where
is
with
being the (
M
-1)-th nearest neighbour of .
This makes "noise" and "boundary" points being "pulled away" from each other.
Genie++ clustering algorithm (see
gclust
)
with respect to the mutual reachability distance gains the ability to
identify some observations are noise points.
Note that the case M
= 2 corresponds to the original distance, but we
determine the 1-nearest neighbours separately as well, which is a bit
suboptimal; you can file a feature request if this makes your data analysis
tasks too slow.
Matrix of class mst
with n-1 rows and 3 columns:
from
, to
and dist
. It holds from
< to
.
Moreover, dist
is sorted nondecreasingly.
The i-th row gives the i-th edge of the MST.
(from[i], to[i])
defines the vertices (in 1,...,n)
and dist[i]
gives the weight, i.e., the
distance between the corresponding points.
The method
attribute gives the name of the distance used.
The Labels
attribute gives the labels of all the input points.
If M
> 1, the nn
attribute gives the indices of the M
-1
nearest neighbours of each point.
Marek Gagolewski and other contributors
Jarnik V., O jistem problemu minimalnim, Prace Moravske Prirodovedecke Spolecnosti 6, 1930, 57-63.
Olson C.F., Parallel algorithms for hierarchical clustering, Parallel Comput. 21, 1995, 1313-1325.
Prim R., Shortest connection networks and some generalisations, Bell Syst. Tech. J. 36, 1957, 1389-1401.
Campello R.J.G.B., Moulavi D., Sander J., Density-based clustering based on hierarchical density estimates, Lecture Notes in Computer Science 7819, 2013, 160-172, doi:10.1007/978-3-642-37456-2_14.
The official online manual of genieclust at https://genieclust.gagolewski.com/
Gagolewski M., genieclust: Fast and robust hierarchical clustering, SoftwareX 15:100722, 2021, doi:10.1016/j.softx.2021.100722.
emst_mlpack()
for a very fast alternative
in case of (very) low-dimensional Euclidean spaces (and M
= 1).
library("datasets") data("iris") X <- iris[1:4] tree <- mst(X)
library("datasets") data("iris") X <- iris[1:4] tree <- mst(X)