The aims of this exercise are to learn how to use R to estimate diversity indices and species accumulation curves. You will need some of these functions to complete your paleoecology assessment. We will be using some made up data about Pokemon sightings within the Museum.
2.1 Before you start
- Open the
02-DiversityIndices.RProj
file in the02-DiversityIndices
folder to open your R Project for this exercise. - Make yourself a new R script for your code.
You will also need to install the following packages:
vegan
picante
knitr
2.2 Preparation
To begin we need to load the packages for this practical.
# Load the packageslibrary(vegan)library(picante)library(knitr)
2.3 What are diversity indices?
A diversity index is a measure of the “diversity” of an area. Diversity could be measured in terms of the numbers of species (or higher taxonomic groupings like genera, families, phyla), or other metrics such as number of haplotypes if you’re interested in genetic diversity, or number of functional groups for studies of functional diversity.
Many diversity indices also account for how evenly spread these different types are.For example, they identify whether there are there five species with 10 individuals of each (even), or five species with one species with 96 individuals and four species with one individual each (uneven). These differences in evenness may be as important for ecosystem function as the number of species.
There are about a million different diversity indices (OK this is a slight exaggeration but there are a lot, check out the vegan package vignette and search for “diversity”), and different people prefer different measures for different questions. Amusingly the vegan
help file for the function diversify
states “these indices are all very closely related Hill (1973), and there is no reason to despise one more than others (but if you are a graduate student, don’t drag me in, but obey your Professor’s orders)”.
2.3.1 \(\alpha\), \(\beta\) and \(\gamma\) diversity
These concepts were originally proposed by Whittaker (1960) and expanded in Whittaker (1972).
- \(\alpha\) (alpha) diversity is the mean species diversity in sites or habitats at a local scale.
- \(\beta\) (beta) diversity is turnover in \(\alpha\) diversity among different sites.
- \(\gamma\) (gamma) diversity is diversity at the landscape level.
For example, if we count the species in Hyde Park and Green Park, we’d have a measure of \(\alpha\) diversity for each. \(\beta\) diversity would measure the difference between species found in Hyde Park and those found in Green Park. \(\gamma\) diversity would be all the species found across London. \(\beta\) diversity is the hardest to measure, and there are far more metrics for measuring this than any others.
2.4 Practical example using Pokemon…
We’re going to play around with some invented data on sampling sites within the Museum and the Pokemon we’ve managed to find there (don’t complain about my unlikely Pokemon combinations, they’re just made up data with Pokemon names rather than A, B, C etc!).
Who doesn’t love a practical handout being interrupted by an enormous Pikachu?
First read in the data and take a look at it.
# Read in the datasetpokemon <- read.csv("data/pokemon-communities.csv")
# Look at the data# I used kable from the knitr package so it makes a nice neat tablekable(pokemon)
Site | Abundance | Species |
---|---|---|
site01 | 5 | Bulbasaur |
site01 | 3 | Charmander |
site02 | 6 | Pidgey |
site02 | 6 | Rattata |
site02 | 1 | Spearow |
site03 | 1 | Bulbasaur |
site03 | 1 | Charmander |
site03 | 1 | Pidgey |
site03 | 10 | Rattata |
site03 | 1 | Spearow |
site03 | 1 | Pikachu |
site04 | 1 | Pikachu |
site04 | 2 | Charmander |
site05 | 3 | Squirtle |
site05 | 2 | Caterpie |
site05 | 4 | Pidgey |
site05 | 3 | Rattata |
site06 | 1 | Psyduck |
site07 | 3 | Bulbasaur |
site07 | 3 | Charmander |
site07 | 3 | Squirtle |
site07 | 3 | Caterpie |
site07 | 3 | Pidgey |
site07 | 3 | Rattata |
site07 | 3 | Spearow |
site08 | 5 | Bulbasaur |
site08 | 5 | Caterpie |
site09 | 10 | Squirtle |
site10 | 3 | Pidgey |
site10 | 6 | Charmander |
site10 | 2 | Zubat |
For the vegan
functions to work you need a matrix where the columns are the species names, the rows are the communities, and the contents of the matrix are the numbers of each species found at each site (or presence absence data as 0s and 1s). We can use the sample2matrix
function in picante
to do this easily.
Note that this only works if your first variable is the name of the site, your second is abundance and your third is the names of the species present.
# Create a matrix we can use with veganpokemon.matrix <- sample2matrix(pokemon)# Look at the matrixkable(pokemon.matrix)
Bulbasaur | Caterpie | Charmander | Pidgey | Pikachu | Psyduck | Rattata | Spearow | Squirtle | Zubat | |
---|---|---|---|---|---|---|---|---|---|---|
site01 | 5 | 0 | 3 | 0 | 0 | 0 | 0 | 0 | 0 | 0 |
site02 | 0 | 0 | 0 | 6 | 0 | 0 | 6 | 1 | 0 | 0 |
site03 | 1 | 0 | 1 | 1 | 1 | 0 | 10 | 1 | 0 | 0 |
site04 | 0 | 0 | 2 | 0 | 1 | 0 | 0 | 0 | 0 | 0 |
site05 | 0 | 2 | 0 | 4 | 0 | 0 | 3 | 0 | 3 | 0 |
site06 | 0 | 0 | 0 | 0 | 0 | 1 | 0 | 0 | 0 | 0 |
site07 | 3 | 3 | 3 | 3 | 0 | 0 | 3 | 3 | 3 | 0 |
site08 | 5 | 5 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 |
site09 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 10 | 0 |
site10 | 0 | 0 | 6 | 3 | 0 | 0 | 0 | 0 | 0 | 2 |
2.5 Species diversity indices
2.5.1 \(\alpha\) diversity
The simplest measure of \(\alpha\) diversity is just the number of species in each site.You can easily extract this as follows.
specnumber(pokemon.matrix)
## site01 site02 site03 site04 site05 site06 site07 site08 site09 site10 ## 2 3 6 2 4 1 7 2 1 3
Simpson’s and Shannon’s diversity indices can be estimated using the function diversity
.
# Simpson's indexdiversity(pokemon.matrix, index = "simpson")
## site01 site02 site03 site04 site05 site06 site07 ## 0.4687500 0.5680473 0.5333333 0.4444444 0.7361111 0.0000000 0.8571429 ## site08 site09 site10 ## 0.5000000 0.0000000 0.5950413
# Shannon's indexdiversity(pokemon.matrix, index = "shannon")
## site01 site02 site03 site04 site05 site06 site07 ## 0.6615632 0.9110175 1.1729935 0.6365142 1.3579779 0.0000000 1.9459101 ## site08 site09 site10 ## 0.6931472 0.0000000 0.9949236
2.5.2 \(\beta\) diversity
The function betadiver
allows you to estimate all the \(\beta\) diversity indices mentioned in Koleff, Gaston, and Lennon (2003). For help on which indices are included type:
betadiver(help=TRUE)
## 1 "w" = (b+c)/(2*a+b+c)## 2 "-1" = (b+c)/(2*a+b+c)## 3 "c" = (b+c)/2## 4 "wb" = b+c## 5 "r" = 2*b*c/((a+b+c)^2-2*b*c)## 6 "I" = log(2*a+b+c) - 2*a*log(2)/(2*a+b+c) - ((a+b)*log(a+b) +## (a+c)*log(a+c)) / (2*a+b+c)## 7 "e" = exp(log(2*a+b+c) - 2*a*log(2)/(2*a+b+c) - ((a+b)*log(a+b)## + (a+c)*log(a+c)) / (2*a+b+c))-1## 8 "t" = (b+c)/(2*a+b+c)## 9 "me" = (b+c)/(2*a+b+c)## 10 "j" = a/(a+b+c)## 11 "sor" = 2*a/(2*a+b+c)## 12 "m" = (2*a+b+c)*(b+c)/(a+b+c)## 13 "-2" = pmin(b,c)/(pmax(b,c)+a)## 14 "co" = (a*c+a*b+2*b*c)/(2*(a+b)*(a+c))## 15 "cc" = (b+c)/(a+b+c)## 16 "g" = (b+c)/(a+b+c)## 17 "-3" = pmin(b,c)/(a+b+c)## 18 "l" = (b+c)/2## 19 "19" = 2*(b*c+1)/(a+b+c)/(a+b+c-1)## 20 "hk" = (b+c)/(2*a+b+c)## 21 "rlb" = a/(a+c)## 22 "sim" = pmin(b,c)/(pmin(b,c)+a)## 23 "gl" = 2*abs(b-c)/(2*a+b+c)## 24 "z" = (log(2)-log(2*a+b+c)+log(a+b+c))/log(2)
Note that some of these are similarity indices, and some are dissimilarity indices.See Koleff, Gaston, and Lennon (2003) for more details. Two commonly used similarity indices are Jaccard’s index and Sorenson’s index which can be estimated as follows (note that completely different communities get a score of 0):
# Jaccard's indexbetadiver(pokemon.matrix, method = "j")
## site01 site02 site03 site04 site05 site06## site02 0.0000000 ## site03 0.3333333 0.5000000 ## site04 0.3333333 0.0000000 0.3333333 ## site05 0.0000000 0.4000000 0.2500000 0.0000000 ## site06 0.0000000 0.0000000 0.0000000 0.0000000 0.0000000 ## site07 0.2857143 0.4285714 0.6250000 0.1250000 0.5714286 0.0000000## site08 0.3333333 0.0000000 0.1428571 0.0000000 0.2000000 0.0000000## site09 0.0000000 0.0000000 0.0000000 0.0000000 0.2500000 0.0000000## site10 0.2500000 0.2000000 0.2857143 0.2500000 0.1666667 0.0000000## site07 site08 site09## site02 ## site03 ## site04 ## site05 ## site06 ## site07 ## site08 0.2857143 ## site09 0.1428571 0.0000000 ## site10 0.2500000 0.0000000 0.0000000
# Shannon's indexbetadiver(pokemon.matrix, method = "sor")
## site01 site02 site03 site04 site05 site06## site02 0.0000000 ## site03 0.5000000 0.6666667 ## site04 0.5000000 0.0000000 0.5000000 ## site05 0.0000000 0.5714286 0.4000000 0.0000000 ## site06 0.0000000 0.0000000 0.0000000 0.0000000 0.0000000 ## site07 0.4444444 0.6000000 0.7692308 0.2222222 0.7272727 0.0000000## site08 0.5000000 0.0000000 0.2500000 0.0000000 0.3333333 0.0000000## site09 0.0000000 0.0000000 0.0000000 0.0000000 0.4000000 0.0000000## site10 0.4000000 0.3333333 0.4444444 0.4000000 0.2857143 0.0000000## site07 site08 site09## site02 ## site03 ## site04 ## site05 ## site06 ## site07 ## site08 0.4444444 ## site09 0.2500000 0.0000000 ## site10 0.4000000 0.0000000 0.0000000
Note that the outputs here are pairwise matrices, as these indices measure the similarity among each pair of sites. You can estimate Whittaker’s original version using method = "w"
(this is a dissimilarity method so completely different communities get a score of 1).
# Whittaker's betadiversity indexbetadiver(pokemon.matrix, method = "w")
## site01 site02 site03 site04 site05 site06## site02 1.0000000 ## site03 0.5000000 0.3333333 ## site04 0.5000000 1.0000000 0.5000000 ## site05 1.0000000 0.4285714 0.6000000 1.0000000 ## site06 1.0000000 1.0000000 1.0000000 1.0000000 1.0000000 ## site07 0.5555556 0.4000000 0.2307692 0.7777778 0.2727273 1.0000000## site08 0.5000000 1.0000000 0.7500000 1.0000000 0.6666667 1.0000000## site09 1.0000000 1.0000000 1.0000000 1.0000000 0.6000000 1.0000000## site10 0.6000000 0.6666667 0.5555556 0.6000000 0.7142857 1.0000000## site07 site08 site09## site02 ## site03 ## site04 ## site05 ## site06 ## site07 ## site08 0.5555556 ## site09 0.7500000 1.0000000 ## site10 0.6000000 1.0000000 1.0000000
2.5.3 \(\gamma\) diversity
In this example, \(\gamma\) diversity is the total number of species found across all sites. We can very simply calculate this in R using the following code:
# How many unique species are there?length(unique(pokemon$Species[pokemon$Abundance > 0]))
## [1] 10
# To view unique speciesunique(pokemon$Species)
## [1] "Bulbasaur" "Charmander" "Pidgey" "Rattata" "Spearow" ## [6] "Pikachu" "Squirtle" "Caterpie" "Psyduck" "Zubat"
Note that the [pokemon$Abundance > 0]
bit of the code ensures we don’t count species where we have them in the species list, but their abundance at all sites is zero.
2.6 Species accumulation curves (Colwell & Coddington 1994)
Often when we talk about species diversity we’re interested in the total diversity of an area or a site. For example, if we want to conserve a patch of woodland, we might need to know how many species in total live there. Sounds easy enough right?Just go out and sample the heck out of that woodland…
The problem of course is that sampling is time consuming and expensive, and in conservation we don’t have much time or money. In addition, completely sampling all species in an area is difficult, especially for small, rare, shy species. Instead we often estimate species richness by various means. Species accumulation curves are one way to do this.
Species accumulation curves are graphs showing the cumulative number of species recorded in an area or site as a function of the cumulative sampling effort taken to search for them. Sampling effort can be number of quadrats, number of hours of sampling etc. for \(\alpha\) diversity, or number of sites if trying to get an estimate of \(\gamma\) diversity.
The idea is that as you sample more, you should get closer to discovering all the species in the area. The first few samples you take will probably have lots of species you haven’t recorded yet, but this rate of discovery should slow down. Eventually you hope that you’ll stop finding any new species to record so the curve will asymptote, but in reality sampling is rarely that thorough. Luckily we can use species accumulation curves to estimate where the curve would asymptote if we kept on sampling.
2.6.1 Pokemon species accumulation curves
Let’s try this for our Pokemon, how many species might be hiding in the Museum if we sampled thoroughly?
We can use the pokemon.matrix
we’ve already created and estimate the accumulation curve using the vegan
function specaccum
. There are lots of methods for estimating these curves but we will use method = "random"
. This works as follows.
It randomly selects a site and calculates the initial richness, then randomly selects a second site and calculates the cumulative richness (i.e.the second site plus the first site), and repeats this until all the sites have been used. It then repeats this process 1000 times. You can change the number of times it does this using permutations
but 1000 is about right - too few and the curves are not smooth, too many and it takes ages. The function outputs the mean cumulative richness and standard deviations for across all 1000 permutations. We do this because our curve will look different depending on which site we start with, and so will give a different total richness estimate. Randomising the order helps us get a better estimate of the total richness, and the standard error on that estimate.
To do this for our Pokemon:
# Fit species accumulation curvepokemon.curve <- specaccum(pokemon.matrix, method = "random", permutations = 1000)# Look at the resultspokemon.curve
## Species Accumulation Curve## Accumulation method: random, with 1000 permutations## Call: specaccum(comm = pokemon.matrix, method = "random", permutations = 1000) ## ## ## Sites 1.000000 2.000000 3.000000 4.000000 5.000000 6.000000 7.000000## Richness 3.016000 5.155000 6.654000 7.707000 8.422000 8.927000 9.288000## sd 1.866341 1.696424 1.302298 0.920869 0.743282 0.685668 0.627255## ## Sites 8.000000 9.000000 10## Richness 9.563000 9.785000 10## sd 0.531332 0.411028 0
# Plot the curveplot(pokemon.curve, ci.type = "poly", col = "blue", ci.col = "lightblue", lwd = 2, ci.lty = 0, xlab = "number of sites", ylab = "cumulative number of Pokemon species")
"ci.type = "poly"
tells R that you want a shaded area showing the confidence intervals from your randomisations. You can play around with the colours etc. if you want to.
For those of you who prefer to use ggplot, we can plot these curves as follows. Note that because ggplot works with dataframes we first need to create a dataframe from the pokemon.curve
object. Also we use standard deviation * 1.96 to get the confidence intervals.
# Load ggplotlibrary(ggplot2)# Make a new dataframepokemon.curve.df <- data.frame(sites = pokemon.curve$sites, richness = pokemon.curve$richness, sd = pokemon.curve$sd)# Plotggplot(pokemon.curve.df, aes(x = sites, y = richness)) + # Add line geom_line() + # Add confidence intervals geom_ribbon(aes(ymin = richness - 1.96*sd, ymax = richness + 1.96*sd), alpha = 0.5, colour = "grey") + # Remove grey background theme_bw(base_size = 14)
To demonstrate why we need the randomisations, look at two curves for just one permutation each.
# Fit one curve with just one permutationpokemon.curve1 <- specaccum(pokemon.matrix, method = "random", permutations = 1)# Fit another curve with just one permutationpokemon.curve2 <- specaccum(pokemon.matrix, method = "random", permutations = 1)# Set the plotting window so we can plot two plotspar(mfrow = c(1,2))# Plot the first curveplot(pokemon.curve1, xlab = "number of sites", ylab = "cumulative number of Pokemon species")# Plot the second curveplot(pokemon.curve2, xlab = "number of sites", ylab = "cumulative number of Pokemon species")
# Reset the plotting window so we see just one plot againpar(mfrow = c(1,1))
Finally to estimate total species richness across all sites we can (again) use many different metrics. Some common ones include Chao 2 (Chao 1987), Jackknife and Bootstrapping approaches and these are easy to estimate using the vegan
function specpool
.
# Estimate diversityspecpool(pokemon.matrix)
## Species chao chao.se jack1 jack1.se jack2 boot boot.se n## All 10 11.8 3.394113 11.8 1.272792 12.68889 10.90352 0.7502033 10
Estimates range from 10.9 \(\pm\) 0.75 (boot) to 11.8 \(\pm\) 3.39 (chao). So we can be fairly confident there are between 10 and 15 (\(11.8 + 3.39 = 15.19\)) Pokemon in the Museum.
2.7 Practical exercise
In the data folder there is another (invented) dataset using British bats called bat-communities.csv
. Read in the data, manipulate it as required by vegan
, then answer the following questions.
Which site has the fewest species?
How many different species are there in total?
What is Simpson’s diversity index for Site J?
Draw a species accumulation curve for the bats and estimate the total number of species. If you round up numbers with decimal places, what is the maximum number of species estimated by any metric?
References
Chao, Anne. 1987. “Estimating the Population Size for Capture-Recapture Data with Unequal Catchability.” Biometrics, 783–91.
Hill, Mark O. 1973. “Diversity and Evenness: A Unifying Notation and Its Consequences.” Ecology 54 (2): 427–32.
Koleff, Patricia, Kevin J Gaston, and Jack J Lennon. 2003. “Measuring Beta Diversity for Presence–Absence Data.” Journal of Animal Ecology 72 (3): 367–82.
Whittaker, Robert H. 1972. “Evolution and Measurement of Species Diversity.” Taxon 21 (2-3): 213–51.
Whittaker, Robert Harding. 1960. “Vegetation of the Siskiyou Mountains, Oregon and California.” Ecological Monographs 30 (3): 279–338.