Skip to content

Commit ccf1e94

Browse files
committed
added RunningMean Class/Module
allows for fast and easy RunningMean Calculation in C++
1 parent d91c9c9 commit ccf1e94

File tree

5 files changed

+115
-1
lines changed

5 files changed

+115
-1
lines changed

DESCRIPTION

+3
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,11 @@ Imports:
1919
glue,
2020
shiny,
2121
plotly
22+
Depends:
23+
Rcpp (>= 0.12.15)
2224
Remotes: ropenscilabs/webrockets
2325
LinkingTo: Rcpp
26+
RcppModules: RunningMeanModule
2427
VignetteBuilder: knitr
2528
Encoding: UTF-8
2629
RoxygenNote: 6.1.1

R/zzz.R

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,6 @@
11
.onUnload <- function(libpath) {
22
library.dynam.unload("arduinor", libpath)
3-
}
3+
}
4+
5+
Rcpp::loadModule("RunningMeanModule", TRUE)
6+

src/RcppExports.cpp

+3
Original file line numberDiff line numberDiff line change
@@ -65,12 +65,15 @@ BEGIN_RCPP
6565
END_RCPP
6666
}
6767

68+
RcppExport SEXP _rcpp_module_boot_RunningMeanModule();
69+
6870
static const R_CallMethodDef CallEntries[] = {
6971
{"_arduinor_ar_init", (DL_FUNC) &_arduinor_ar_init, 2},
7072
{"_arduinor_ar_is_open", (DL_FUNC) &_arduinor_ar_is_open, 1},
7173
{"_arduinor_ar_close", (DL_FUNC) &_arduinor_ar_close, 1},
7274
{"_arduinor_ar_read", (DL_FUNC) &_arduinor_ar_read, 4},
7375
{"_arduinor_ar_flush", (DL_FUNC) &_arduinor_ar_flush, 1},
76+
{"_rcpp_module_boot_RunningMeanModule", (DL_FUNC) &_rcpp_module_boot_RunningMeanModule, 0},
7477
{NULL, NULL, 0}
7578
};
7679

src/RunningMean.cpp

+50
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
#include "RunningMean.h"
2+
3+
4+
RunningMean::RunningMean() : n(0) {
5+
stop("n (size of window) must be supplied");
6+
}
7+
8+
// Initiates a new Running Mean class of size n (constant)
9+
RunningMean::RunningMean(int n) : n(n) {
10+
if (n <= 0) stop("n (size of window) must be supplied");
11+
deq.resize(n);
12+
n_values = 0;
13+
sum = 0;
14+
}
15+
16+
// inserts a new value into the internal deque. Deletes the oldest value
17+
void RunningMean::insert(double v) {
18+
if (n == 0) stop("n (size of window) must be larger than zero");
19+
20+
// TODO: What happens if we insert a missing value?
21+
if (!R_IsNA(v) && !R_IsNaN(v)) {
22+
n_values++;
23+
sum += v - deq.back();
24+
deq.push_front(v);
25+
deq.pop_back();
26+
}
27+
}
28+
29+
// Returns the running mean of the values in the deque
30+
// if the size is 0, return NA, if the number of already inserted values (n_values)
31+
// is smaller than the size of the deque, use n_values. This allows the running mean
32+
// to return values even if the number of values is smaller than window lengths
33+
// could also be set to NA if needed...
34+
double RunningMean::get_mean() {
35+
if (n == 0) return NA_REAL;
36+
if (n_values < n) return(sum / n_values);
37+
return sum / n;
38+
}
39+
40+
// prints the values of the deque
41+
void RunningMean::show() {
42+
Rcout << "RunningMean object <" << static_cast<const void*>(this) <<
43+
"> window size " << n << ", number of inserts " << n_values << "\n" <<
44+
deq[0];
45+
for (int i = 1; i < n; ++i) {
46+
Rcout << " " << deq[i];
47+
}
48+
Rcout << "\n";
49+
}
50+

src/RunningMean.h

+55
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
#include <Rcpp.h>
2+
using namespace Rcpp;
3+
4+
/*
5+
* Example R Code
6+
*
7+
* # Initiate the Class with a window-length of size 3
8+
* obj <- arduinor:::RunningMean$new(3)
9+
*
10+
* # insert a new value
11+
* obj$insert(1.03)
12+
* obj$insert(1.05)
13+
* obj$insert(1.04)
14+
*
15+
* # inspect the object
16+
* obj
17+
*
18+
* # get the running mean
19+
* obj$get_mean()
20+
*
21+
* # add another value, effectively removing the first insert (1.03)
22+
* obj$insert(1.06)
23+
* obj
24+
* obj$get_mean()
25+
*
26+
*/
27+
28+
// defines the RunningMean class
29+
class RunningMean{
30+
public:
31+
RunningMean();
32+
RunningMean(int n);
33+
34+
void insert(double val);
35+
double get_mean();
36+
void show();
37+
38+
private:
39+
std::deque<double> deq;
40+
const int n;
41+
int n_values;
42+
double sum;
43+
};
44+
45+
// Exposes the class
46+
RCPP_MODULE(RunningMeanModule) {
47+
class_<RunningMean>("RunningMean")
48+
.default_constructor("Default constructor")
49+
.constructor<int>("Constructor with an argument")
50+
.method("insert", &RunningMean::insert)
51+
.method("get_mean", &RunningMean::get_mean)
52+
.method("show", &RunningMean::show)
53+
;
54+
}
55+

0 commit comments

Comments
 (0)