当样本大小和概率变化时,进行有效的多项式抽样

时间:2014-04-16 01:13:03

标签: r performance vectorization multinomial

这个问题涉及来自具有不同样本大小和概率的多项分布的有效采样。下面我描述了我使用的方法,但想知道是否可以通过一些智能矢量化来改进它。

我正在模拟多个群体中的生物传播。人口j中的个人以概率i分散到人口p[i, j]。鉴于人口1中的初始丰度为10,以及分别为{1}}对人口1,2和3的概率,我们可以使用c(0.1, 0.3, 0.6)来模拟分散过程:

rmultinom

我们可以将其扩展为考虑set.seed(1) rmultinom(1, 10, c(0.1, 0.3, 0.6)) # [,1] # [1,] 0 # [2,] 3 # [3,] 7 来源群体:

n

在上面,set.seed(1) n <- 3 p <- replicate(n, diff(c(0, sort(runif(n-1)), 1))) X <- sample(100, n) 是从一个群体(列)移动到另一个群体(行)的概率矩阵,p是初始种群大小的向量。现在可以用以下方式模拟分散在每对种群(以及它们所在的种群)之间的个体数量:

X

其中sapply(seq_len(ncol(p)), function(i) { rmultinom(1, X[i], p[, i]) }) # [,1] [,2] [,3] # [1,] 19 42 11 # [2,] 8 18 43 # [3,] 68 6 8 行和i列的元素值是从人口j移动到人口j的人数。此矩阵的i给出了新的种群大小。

我想用常数概率矩阵重复这一次,但具有不同的(预定义的)初始丰度。以下小例子实现了这一点,但是对于更大的问题效率低下。得到的矩阵在5个模拟中的每一个中给出了三个群体中每个群体的后扩散丰度,其中群体具有不同的初始丰度。

rowSums

有没有办法更好地矢量化这个问题?

3 个答案:

答案 0 :(得分:5)

这是多多项式的RcppGSL实现。但是,它需要您独立安装gsl ....这可能不太实用。

// [[Rcpp::depends(RcppGSL)]]

#include <RcppGSL.h>
#include <gsl/gsl_rng.h>
#include <gsl/gsl_randist.h>
#include <unistd.h>            // getpid

Rcpp::IntegerVector rmn(unsigned int N, Rcpp::NumericVector p, gsl_rng* r){

    size_t K = p.size();

    Rcpp::IntegerVector x(K);
    gsl_ran_multinomial(r, K, N, p.begin(), (unsigned int *) x.begin());
    return x;             // return results vector
}

Rcpp::IntegerVector gsl_mmm_1(Rcpp::IntegerVector N, Rcpp::NumericMatrix P, gsl_rng* r){
    size_t K = N.size();
    int i;
    Rcpp::IntegerVector x(K);
    for(i=0; i<K; i++){
        x += rmn(N[i], P(Rcpp::_, i), r);
    }
    return x;
}

// [[Rcpp::export]]
Rcpp::IntegerMatrix gsl_mmm(Rcpp::IntegerMatrix X_, Rcpp::NumericMatrix P){
    int j;
    gsl_rng * r = gsl_rng_alloc (gsl_rng_mt19937);
    long seed = rand()/(((double)RAND_MAX + 1)/10000000) * getpid();
    gsl_rng_set (r, seed);
    Rcpp::IntegerMatrix X(X_.nrow(), X_.ncol());
    for(j=0; j<X.ncol(); j++){
        X(Rcpp::_, j) = gsl_mmm_1(X_(Rcpp::_,j), P, r);
    }
    gsl_rng_free (r);
    return X;
}

我还将它与纯R实现和jbaums的版本

进行比较
library(Rcpp)
library(microbenchmark)
sourceCpp("gsl.cpp")

P = matrix(c(c(0.1,0.2,0.7),c(0.3,0.3,0.4),c(0.5,0.3,0.2)),nc=3)
X = matrix(c(c(30,40,30),c(20,40,40)), nc=2)

mmm = function(X, P){
    n = ncol(X)
    p = nrow(X)
    Reduce("+", lapply(1:p, function(j) {
        Y = matrix(0,p,n)
        for(i in 1:n) Y[,i] = rmultinom(1, X[j,i], P[,j])
        Y
    }))
}

jbaums = function(X,P){
    apply(sapply(apply(X, 2, function(x) {
      lapply(seq_len(ncol(P)), function(i) {
        rmultinom(1, x[i], P[, i])
      })
    }), function(x) do.call(cbind, x), simplify='array'), nrow(X), rowSums)
}
microbenchmark(jbaums(X,P), mmm(X,P), gsl_mmm(X, P))

这是结果

> microbenchmark(jbaums(X,P), mmm(X,P), gsl_mmm(X, P))
Unit: microseconds
          expr     min       lq  median       uq     max neval
  jbaums(X, P) 165.832 172.8420 179.185 187.2810 339.280   100
     mmm(X, P)  60.071  63.5955  67.437  71.5775  92.963   100
 gsl_mmm(X, P)  10.529  11.8800  13.671  14.6220  40.857   100

gsl版本比纯R版本快6倍。

答案 1 :(得分:1)

例如:

# make the example in Rcpp you mention:
library(Rcpp)
library(inline)
src <- 'Environment stats("package:stats");
Function rmultinom = stats["rmultinom"];
NumericVector some_p(1000, 1.0/1000);
return(rmultinom(1,1, some_p));'

fx <- rcpp(signature(), body=src)

# now compare the two
library(rbenchmark)
benchmark(fx(),rmultinom(1,1,c(1000,1/1000)),replications=10000)

#                            test replications elapsed relative user.self sys.self user.child sys.child
#    1                       fx()        10000   1.126   13.901     1.128        0          0         0
#    2 rmultinom(1, 1, c(1/1000))        10000   0.081    1.000     0.080        0          0         0

答案 2 :(得分:1)

我发现BH包会将boost个库带到表中。这样可以生成以下内容,它产生与@ RandyLai gsl_mmm相同的输出以及上面问题中的代码。 (我相信启用c ++ 11支持应该random可以使用BH。)

// [[Rcpp::depends(BH)]]
#include <Rcpp.h>

#include <boost/random.hpp>
#include <boost/random/mersenne_twister.hpp>
#include <boost/random/discrete_distribution.hpp>

using namespace Rcpp;

typedef boost::mt19937 RNGType;
RNGType rng(123);


NumericVector rowSumsC(IntegerMatrix x) {
  int nrow = x.nrow(), ncol = x.ncol();
  IntegerVector out(nrow);

  for (int i = 0; i < nrow; i++) {
    double total = 0;
    for (int j = 0; j < ncol; j++) {
      total += x(i, j);
    }
    out[i] = total;
  }
  return wrap(out);
}

// [[Rcpp::export]]
IntegerMatrix rmm(IntegerMatrix X, NumericMatrix P) {
  int niter = X.ncol(), nx = X.nrow();
  IntegerMatrix out(nx, niter);
  for (int j = 0; j < niter; j++) {
    IntegerMatrix tmp(nx, nx);
    for (int i = 0; i < nx; i++) {
      for (int n = 0; n < X(i, j); n++) {
        boost::random::discrete_distribution<> dist(P(_, i));
        tmp(dist(rng), i)++;
      }
    }
    out(_, j) = rowSumsC(tmp);
  }
  return out;
}

rowSumsC由@hadley提供,here

然而,在我的机器上,这比Randy的gsl_mmm要慢得多,而且当有很多试验时,确实比我的R版慢。我怀疑这是由于编码效率低下,但是增强discrete_distribution也单独执行每个多项式试验,而这个过程在使用gsl时会出现矢量化。我是c ++的新手,所以不确定这是否可以提高效率。

P <- matrix(c(c(0.1, 0.2, 0.7), c(0.3, 0.3, 0.4), c(0.5, 0.3, 0.2)), nc=3)
X <- matrix(c(c(30, 40, 30), c(20, 40, 40)), nc=2)
library(BH)
microbenchmark(jbaums(X, P), rmm(X, P))

# Unit: microseconds
#          expr     min       lq  median       uq     max neval
#  jbaums(X, P) 124.988 129.5065 131.464 133.8735 348.763   100
#     rmm(X, P)  59.031  60.0850  62.043  62.6450 117.459   100