Domanda

Volevo verificare se c'è un trucco preesistente per na.locf (dal pacchetto zoo), rle e inverse.rle in RCpp?

Ho scritto un anello da implementare, ad es.Ho fatto l'implementazione di na.locf(x, na.rm=FALSE, fromLast=FALSE) come segue:

#include <Rcpp.h>
using namespace Rcpp;

//[[Rcpp::export]]
NumericVector naLocf(NumericVector x) {
  int n=x.size();
  for (int i=1;i<n;i++) {
    if (R_IsNA(x[i]) & !R_IsNA(x[i-1])) {
      x[i]=x[i-1];
    }
  }
  return x;
}
.

Mi stavo solo chiedendo che da quando queste sono funzioni di base, qualcuno potrebbe averle già implementato in RCpp in un modo migliore (può essere evitato il loop) o un modo più veloce?

È stato utile?

Soluzione

L'unica cosa che direi è che stai provando per NA due volte per ogni valore quando è necessario farlo solo una volta.Test per NA non è un funzionamento gratuito.Forse qualcosa del genere:

//[[Rcpp::export]]
NumericVector naLocf(NumericVector x) {
    int n = x.size() ;
    double v = x[0]
    for( int i=1; i<n; i++){
        if( NumericVector::is_na(x[i]) ) {
            x[i] = v ;
        } else {
            v = x[i] ;    
        }
    }

    return x;
}
.

Ciò tuttavia tuttavia fa cose inutili, come impostare v ogni volta in cui potremmo farlo solo per l'ultima volta che non vediamo NA.Possiamo provare qualcosa del genere:

//[[Rcpp::export]]
NumericVector naLocf3(NumericVector x) {
    double *p=x.begin(), *end = x.end() ;
    double v = *p ; p++ ;

    while( p < end ){
        while( p<end && !NumericVector::is_na(*p) ) p++ ;
        v = *(p-1) ;
        while( p<end && NumericVector::is_na(*p) ) {
            *p = v ;
            p++ ;
        }
    }

    return x;
}
.

Ora, possiamo provare alcuni benchmark:

x <- rnorm(1e6)
x[sample(1:1e6, 1000)] <- NA 
require(microbenchmark)
microbenchmark( naLocf1(x), naLocf2(x), naLocf3(x) )
#  Unit: milliseconds
#       expr      min       lq   median       uq      max neval
# naLocf1(x) 6.296135 6.323142 6.339132 6.354798 6.749864   100
# naLocf2(x) 4.097829 4.123418 4.139589 4.151527 4.266292   100
# naLocf3(x) 3.467858 3.486582 3.507802 3.521673 3.569041   100
.

Autorizzato sotto: CC-BY-SA insieme a attribuzione
Non affiliato a StackOverflow
scroll top