## Install extra packages:
## - khroma: colour schemes
## - folio: datasets
## - tabula: visualization
# install.packages(c("khroma", "folio", "tabula"))
# Load packages
library(kairos)
The matrix seriation problem in archaeology is based on three conditions and two assumptions, which Dunnell (1970) summarizes as follows.
The homogeneity conditions state that all the groups included in a seriation must:
The mathematical assumptions state that the distribution of any historical or temporal class:
Theses assumptions create a distributional model and ordering is accomplished by arranging the matrix so that the class distributions approximate the required pattern. The resulting order is inferred to be chronological.
Reciprocal ranking iteratively rearrange rows and/or columns according to their weighted rank in the data matrix until convergence (Ihm 2005).
For a given incidence matrix \(C\):
These two steps are repeated until convergence. Note that this procedure could enter into an infinite loop.
## Build an incidence matrix with random data
set.seed(12345)
<- sample(c(TRUE, FALSE), 400, TRUE, c(0.6, 0.4))
bin <- matrix(bin, nrow = 20)
incidence1
## Get seriation order on rows and columns
## If no convergence is reached before the maximum number of iterations (100),
## it stops with a warning.
<- seriate_rank(incidence1, margin = c(1, 2), stop = 100))
(indices #> <RankPermutationOrder>
#> Permutation order for matrix seriation:
#> - Row order: 6 15 12 14 2 5 10 8 13 20 16 18 19 7 9 1 3 11 17 4...
#> - Column order: 9 4 2 8 3 1 6 5 16 20 15 14 12 17 13 10 19 7 11 18...
## Permute matrix rows and columns
<- permute(incidence1, indices)
incidence2
## Plot matrix
::plot_heatmap(incidence1) +
tabula::scale_fill_logical()
khroma::plot_heatmap(incidence2) +
tabula::scale_fill_logical() khroma
The positive difference from the column mean percentage (in french “écart positif au pourcentage moyen”, EPPM) represents a deviation from the situation of statistical independence (Desachy 2004). As independence can be interpreted as the absence of relationships between types and the chronological order of the assemblages, EPPM is a useful graphical tool to explore significance of relationship between rows and columns related to seriation (Desachy 2004).
## Replicates Desachy 2004
data("compiegne", package = "folio")
## Plot frequencies and EPPM values
::seriograph(compiegne) tabula
## Get seriation order for columns on EPPM using the reciprocal ranking method
## Expected column order: N, A, C, K, P, L, B, E, I, M, D, G, O, J, F, H
<- seriate_rank(compiegne, EPPM = TRUE, margin = 2))
(indices #> <RankPermutationOrder>
#> Permutation order for matrix seriation:
#> - Row order: 1 2 3 4 5...
#> - Column order: 14 1 3 11 16 12 2 5 9 13 4 7 15 10 6 8...
## Permute columns
<- permute(compiegne, indices)
compiegne_permuted
## Plot frequencies and EPPM values
::seriograph(compiegne_permuted) tabula
Correspondence Analysis (CA) is an effective method for the seriation of archaeological assemblages. The order of the rows and columns is given by the coordinates along one dimension of the CA space, assumed to account for temporal variation. The direction of temporal change within the correspondence analysis space is arbitrary: additional information is needed to determine the actual order in time.
## Get data
data("zuni", package = "folio")
## Ford diagram
::plot_ford(zuni) +
tabula::theme(axis.text.y = ggplot2::element_blank()) ggplot2
## Get row permutations from CA coordinates
<- seriate_average(zuni, margin = c(1, 2)))
(zun_indices #> <AveragePermutationOrder>
#> Permutation order for matrix seriation:
#> - Row order: 372 387 350 367 110 417 364 407 357 160 373 406 35...
#> - Column order: 18 14 17 16 13 15 9 8 12 11 6 7 5 10 4 2 3 1...
## Plot CA results
::plot_rows(zun_indices) +
dimensio::labs(title = "Rows") +
ggplot2::theme_bw() ggplot2
## Permute data matrix
<- permute(zuni, zun_indices)
zuni_permuted
## Ford diagram
::plot_ford(zuni_permuted) +
tabula::theme(axis.text.y = ggplot2::element_blank()) ggplot2
Peeples and Schachner (2012) propose a
procedure to identify samples that are subject to sampling error or
samples that have underlying structural relationships and might be
influencing the ordering along the CA space. This relies on a partial
bootstrap approach to CA-based seriation where each sample is replicated
n
times. The maximum dimension length of the convex hull
around the sample point cloud allows to remove samples for a given
cutoff
value.
According to Peeples and Schachner (2012), “[this] point removal procedure [results in] a reduced dataset where the position of individuals within the CA are highly stable and which produces an ordering consistent with the assumptions of frequency seriation.”
## Partial bootstrap CA
## Warning: this may take a few seconds!
<- dimensio::bootstrap(zun_indices, n = 30)
zuni_boot
## Plot convex hull
## Bootstrap CA results for the rows
|>
zuni_boot ::plot_rows(active = FALSE, colour = "obs", fill = "obs") +
dimensio::stat_hull(alpha = 0.2) +
dimensio::labs(title = "Rows") +
ggplot2::theme_bw() +
ggplot2::scale_colour_highcontrast() +
khroma::scale_fill_highcontrast()
khroma
## Bootstrap CA results for the columns
|>
zuni_boot ::plot_columns(colour = "group", fill = "group") +
dimensio::stat_hull(alpha = 0.5) +
dimensio::labs(title = "Columns") +
ggplot2::theme_bw() +
ggplot2::scale_colour_discreterainbow() +
khroma::scale_fill_discreterainbow() khroma
## Replicates Peeples and Schachner 2012 results
## Samples with convex hull maximum dimension length greater than the cutoff
## value will be marked for removal.
## Define cutoff as one standard deviation above the mean
<- function(x) { mean(x) + sd(x) }
fun <- seriate_refine(zun_indices, cutoff = fun, margin = 1))
(zuni_refine #> <RefinePermutationOrder>
#> Permutation order for matrix seriation:
#> - Row order: 372 350 387 367 110 417 364 357 407 356 160 344 34...
#> - Column order: 17 18 14 16 13 15 9 8 12 11 6 10 7 5 4 2 3 1...
#> Partial bootstrap refinement:
#> - Cutoff value: 1.77
#> - Rows to keep: 360 of 420 (86%)
## Plot CA results for the rows
::plot_rows(zuni_refine, colour = "observation", fill = "observation") +
dimensio::labs(title = "Rows") +
ggplot2::theme_bw() +
ggplot2::scale_colour_discreterainbow() +
khroma::scale_fill_discreterainbow() khroma
## Histogram of convex hull maximum dimension length
<- data.frame(length = zuni_refine[["length"]])
hull_length ::ggplot(data = hull_length) +
ggplot2::aes(x = length) +
ggplot2::geom_histogram(breaks = seq(0, 4.5, by = 0.5), fill = "grey70") +
ggplot2::geom_vline(xintercept = zuni_refine[["cutoff"]], colour = "red") +
ggplot2::labs(
ggplot2title = "Convex hull max. dim.",
x = "Maximum length",
y = "Count"
+
) ::theme_bw() ggplot2
## Permute data matrix
<- permute(zuni, zuni_refine)
zuni_permuted2
## Ford diagram
::plot_ford(zuni_permuted2) +
tabula::theme(axis.text.y = ggplot2::element_blank()) ggplot2