在调用已编译的 ODE 以通过 R 包解决时,我正在努力解决一个可能很小的问题,'deSolve'
并且我寻求更多专家用户的建议。
背景
我有几个需要解决的 ODE 系统'deSolve'
。我已经在单独的 C++ 函数(每个模型一个)中定义了 ODE,我通过 R 结合'Rcpp'
. 如果函数从另一个模型获取输入(所以基本上有级联),系统的初始值会发生变化。
这很好用,但是,对于一个模型,我必须为t < 2
. 我尝试在 C++ 函数中执行此操作,但它似乎不起作用。
运行代码示例
#include <Rcpp.h>
using namespace Rcpp;
// [[Rcpp::export("set_ODE")]]
SEXP set_ODE(double t, NumericVector state, NumericVector parameters) {
List dn(3);
double tau2 = parameters["tau2"];
double Ae2_4 = parameters["Ae2_4"];
double d2 = parameters["d2"];
double N2 = parameters["N2"];
double n2 = state["n2"];
double m4 = state["m4"];
double ne = state["ne"];
// change starting conditions for t < 2
if(t < 2) {
n2 = (n2 * m4) / N2;
m4 = n2;
ne = 0;
}
dn[0] = n2*d2 - ne*Ae2_4 - ne/tau2;
dn[1] = ne/tau2 - n2*d2;
dn[2] = -ne*Ae2_4;
return(Rcpp::List::create(dn));
}
/*** R
state <- c(ne = 10, n2 = 0, m4 = 0)
parameters <- c(N2 = 5e17, tau2 = 1e-8, Ae2_4 = 5e3, d2 = 0)
results <- deSolve::lsoda(
y = state,
times = 1:10,
func = set_ODE,
parms = parameters
)
print(results)
*/
输出读取(这里只有前两行):
time ne n2 m4
1 1 1.000000e+01 0.000000e+00 0.000000e+00
2 2 1.000000e+01 2.169236e-07 -1.084618e-11
以防万一:如何运行此代码示例?
我的示例使用RStudio进行了测试:
- 将代码复制到以 *.cpp 结尾的文件中
- 单击“来源”按钮(或
<shift>
+<cmd>
+<s>
)
它在没有 RStudio 的情况下也应该可以工作,但是必须安装包'Rcpp'
和'deSolve'
编译它需要在 Windows 上使用 Rtools、在 Linux 上使用 GNU 编译器和在 macOS 上使用 Xcode 的代码。
问题
据我了解,ne
应该是0
for time = 1
(或t < 2
)。不幸的是,求解器似乎没有考虑我在 C++ 函数中提供的内容,除了 ODE。但是,如果我将state
R 更改为另一个值,它会起作用。不知何故,我在 C++ 中定义的 if 条件被忽略了,但我不明白为什么以及如何计算 C++ 而不是 R 中的初始值。