-
Notifications
You must be signed in to change notification settings - Fork 97
/
Copy pathspecpool.R
83 lines (83 loc) · 2.77 KB
/
specpool.R
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
`specpool` <-
function (x, pool, smallsample = TRUE)
{
x <- as.matrix(x)
if (!(is.numeric(x) || is.logical(x)))
stop("input data must be numeric")
if (missing(pool))
pool <- rep("All", nrow(x))
## check dims
if (length(pool) != NROW(x))
stop("length of 'pool' and number rows in 'x' do not match")
## remove missing values
if (any(nas <- is.na(pool))) {
pool <- pool[!nas]
x <- x[!nas, , drop = FALSE]
}
out <- seq(1:nrow(x))
groups <- table(pool)
inds <- names(groups)
S <- var.chao <- chao <- var.jack1 <- jack.1 <- jack.2 <- var.boot <- bootS <-
rep(NA, length(inds))
names(S) <- names(var.chao) <- names(chao) <- names(var.jack1) <-
names(jack.1) <- names(jack.2) <- names(var.boot) <- names(bootS) <- inds
for (is in inds) {
a1 <- a2 <- NA
gr <- out[pool == is]
n <- length(gr)
if (n <= 0)
next
if (smallsample)
ssc <- (n-1)/n
else
ssc <- 1
X <- x[gr, , drop = FALSE]
freq <- colSums(X > 0)
p <- freq[freq > 0]/n
S[is] <- sum(freq > 0)
if (S[is] == 0)
next
if (n >= 1L)
a1 <- sum(freq == 1)
if (n >= 2L)
a2 <- sum(freq == 2)
else
a2 <- 0
chao[is] <- S[is] + if(!is.na(a2) && a2 > 0)
ssc * a1 * a1/2/a2
else
ssc * a1 * (a1-1)/2
jack.1[is] <- S[is] + a1 * (n - 1)/n
if (n > 1L)
jack.2[is] <- S[is] + a1 * (2 * n - 3)/n -
a2 * (n - 2)^2/n/(n - 1)
else
jack.2[is] <- S[is]
bootS[is] <- S[is] + sum((1 - p)^n)
aa <- if (!is.na(a2) && a2 > 0) a1/a2
else 0
if (!is.na(a2) && a2 > 0)
var.chao[is] <- a1 * ssc * (0.5 + ssc * (1 + aa/4) * aa) * aa
else
var.chao[is] <-
ssc * (ssc * (a1*(2*a1-1)^2/4 - a1^4/chao[is]/4) + a1*(a1-1)/2)
if (!is.na(a1) && a1 > 0) {
jf <- table(rowSums(X[, freq == 1, drop = FALSE] >
0))
var.jack1[is] <- (sum(as.numeric(names(jf))^2 * jf) -
a1/n) * (n - 1)/n
} else {
var.jack1[is] <- 0
}
pn <- (1 - p)^n
X <- X[, freq > 0, drop = FALSE]
Zp <- (crossprod(X == 0)/n)^n - outer(pn, pn, "*")
var.boot[is] <- sum(pn * (1 - pn)) + 2 * sum(Zp[lower.tri(Zp)])
}
out <- list(Species = S, chao = chao, chao.se = sqrt(var.chao),
jack1 = jack.1, jack1.se = sqrt(var.jack1), jack2 = jack.2,
boot = bootS, boot.se = sqrt(var.boot), n = as.vector(groups))
out <- as.data.frame(out)
attr(out, "pool") <- pool
out
}