- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我必须通过引用或指针将数组传递给其他函数,我不在乎,只要它运行速度快即可。这就是我开始使用boost库的原因。我是通过以下方式做到的:
using namespace boost;
typedef multi_array<long double, 4> array_type;
typedef multi_array<long double, 2> twod_array_type;
typedef multi_array<long double, 1> vec_type;
作为函数:
void pde_3d_7_stencil_discretization(array_type& A, vec_type& b, vec_type& x,const int& xdim, const int& ydim,const int& zdim)
void gmressolver3d(array_type& A, vec_type& x, vec_type& rhs,const int& KrylovDim,const int& xdim,const int& ydim,const int& zdim,const int& COP, const int& threeDStencil)
在主函数中:
array_type A(extents[threeDimStencil][COP][COP][xdim*ydim*zdim]);
vec_type b(extents[xdim*ydim*zdim*COP]);
vec_type x(extents[xdim*ydim*zdim*COP]);
pde_3d_7_stencil_discretization(A,b,x,xdim,ydim,zdim);
gmressolver3d(A,x,b,KrylovDim,xdim,ydim,zdim,COP,threeDimStencil);
显然,我做错了什么,因为代码的运行速度确实比静态版本慢,静态版本不涉及任何引用/指针,只是将数组从一个函数传递到另一个函数。
我可以做些什么来 boost 这个过程?
感谢您的任何帮助..
编辑:我发布了这些代码的作用,来自 GMRES 求解器的序列:其中的所有数组也使用 Boost 进行了初始化,例如:
vec_type pp(extents[zdim*xdim*ydim*COP]);
vec_type ppp(extents[zdim*xdim*ydim*COP]);
vec_type w(extents[zdim*xdim*ydim*COP]);
vec_type y(extents[KrylovDim]);
vec_type vv(extents[zdim*xdim*ydim*COP]);
vec_type b(extents[KrylovDim+1]);
vec_type ro(extents[zdim*xdim*ydim*COP]);
vec_type out1(extents[xdim*zdim*ydim*COP]);
vec_type m_jac(extents[xdim*zdim*ydim*COP]);
twod_array_type h(extents[KrylovDim+1][KrylovDim]);
twod_array_type v(extents[zdim*xdim*ydim*COP][KrylovDim]);
twod_array_type hess(extents[KrylovDim+1][KrylovDim]);
array_type maa(extents[threeDStencil][COP][COP][zdim*xdim*ydim]);
array_type maaa(extents[threeDStencil][COP][COP][zdim*xdim*ydim]);
for (i=0;i<m+1;i++){
b[i] = 0;
for(k=0;k<m;k++){
h[i][k] = 0.0;
}
}
for (i=0;i<n;i++){
v[i][0] = ro[i]/r;
}
for(j=0;j<m;j++){
b[0] = r;
vector_zero_fill(n,ppp);
for(i=0;i<n;i++){
vv[i]=v[i][j];
}
//********************MATRIX FREE********************
matrix_vector_product_heptadiagonal_discret(A,vv,pp,xdim,ydim,zdim);
//two_vector_dot_product(n,pp,m_jac);
// if(isPrec)
// forback(A,pp);
//********************MATRIX FREE********************
//pretty fast**
for(i=0;i<=j;i++){
for(k=0;k<n;k++){
h[i][j] = h[i][j] + pp[k]*v[k][i];
}
}
for(i=0;i<=j;i++){
for(k=0;k<n;k++){
ppp[k] = ppp[k] + h[i][j]*v[k][i];
}
}
p=0.0;
for(i=0;i<n;i++){
w[i] = pp[i] - ppp[i];
p = p + pow(w[i],2);
}
h[j+1][j] = sqrt(p);
for(i=0;i<=j+1;i++){
for(k=0;k<=j;k++){
hess[i][k] = h[i][k];
}
}
for(i=0;i<j+1;i++){
c = hess[i][i]/sqrt(pow(hess[i][i],2)+pow(hess[i+1][i],2));
s = hess[i+1][i]/sqrt(pow(hess[i][i],2)+pow(hess[i+1][i],2));
for (k=0;k<=j;k++){
inner1=c*hess[i][k]+s*hess[i+1][k];
inner2=(-s)*hess[i][k]+c*hess[i+1][k];
hess[i][k] = inner1;
hess[i+1][k] = inner2;
}
b[i+1] = -s*b[i];
b[i] = c*b[i];
}
最佳答案
在对 multi_aray 进行零初始化的地方,您可以尝试使用 std::memset
代替。例如
std::memset(b.data(), 0, size_of_b_in_bytes);
您的代码中有几个地方多次索引同一个 multi_array
元素。例如,代替
h[i][j] = h[i][j] + pp[k]*v[k][i]
尝试
h[i][j] += pp[k]*v[k][i]
通常,优化器会自动为您进行此类替换,但对于 multi_array
可能不会。
我还发现了两个可以合并为一个的 for 循环,以避免多次索引同一个 multi_array 元素:
/*
for(i=0; i<=j; i++)
{
for(k=0; k<n; k++)
{
h[i][j] = h[i][j] + pp[k]*v[k][i];
}
}
for(i=0; i<=j; i++)
{
for(k=0; k<n; k++)
{
ppp[k] = ppp[k] + h[i][j]*v[k][i];
}
}
*/
for(i=0; i<=j; i++)
{
for(k=0; k<n; k++)
{
long double& h_elem = h[i][j];
long double v_elem = v[k][i];
h_elem += pp[k]*v_elem;
ppp[k] += h_elem*v_elem;
}
}
可能还有更多类似的。请注意使用引用和变量来“记住”一个元素并避免重新计算它在 multi_array 中的位置。
在代码的最后一个 for 循环中,您可以通过使用临时变量和引用来避免多次重新计算 multi_array 索引:
/*
for(i=0;i<j+1;i++){
c = hess[i][i]/sqrt(pow(hess[i][i],2)+pow(hess[i+1][i],2));
s = hess[i+1][i]/sqrt(pow(hess[i][i],2)+pow(hess[i+1][i],2));
for (k=0;k<=j;k++){
inner1=c*hess[i][k]+s*hess[i+1][k];
inner2=(-s)*hess[i][k]+c*hess[i+1][k];
hess[i][k] = inner1;
hess[i+1][k] = inner2;
}
b[i+1] = -s*b[i];
b[i] = c*b[i];
}
*/
for(i=0;i<j+1;i++){
long double hess_i_i = hess[i][i];
long double hess_ip1_i = hess[i+1][i];
long double temp = sqrt(pow(hess_i_i,2)+pow(hess_ip1_i,2));
c = hess_i_i/temp;
s = hess_ip1_i/temp;
for (k=0;k<=j;k++){
long double& hess_i_k = hess[i][k];
long double& hess_ip1_k = hess[i+1][k];
inner1=c*hess_i_k+s*hess_ip1_k;
inner2=(-s)*hess_i_k+c*hess_ip1_k;
hess_i_k = inner1;
hess_ip1_k = inner2;
}
long double b_i& = b[i];
b[i+1] = -s*b_i;
b_i = c*b_i;
}
仔细检查我的工作 - 可以肯定我在某处犯了错误。请注意,我已将 sqrt(pow(hess_i_i,2)+pow(hess_ip1_i,2))
存储在一个变量中,这样它就不会不必要地计算两次。
我怀疑这些微小的调整能否将运行时间缩短至 5 秒。 multi_array
的问题在于数组维度仅在运行时已知。支持行优先/列优先排序可能还会产生一些开销。
使用 C 风格的多维数组,维度在编译时已知,因此编译器可以生成“更紧凑”的代码。
通过使用 Boost multi_arrays,您基本上是在牺牲速度以换取灵 active 和便利性。
关于c++ - Boost::multi_array——引用太慢,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5283251/
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: template pass by value or const reference or…? 以下对于将函数
我用相同的参数列表重载了一个运算符两次。但返回类型不同: T& operator()(par_list){blablabla} const T& operator()(par_list){bla
假设我有实现接口(interface) I 的 Activity A。我的 ViewModel 类 (VM) 持有对实现接口(interface) I 的对象的引用: class A extends
PHP 如何解释 &$this ?为什么允许? 我遇到了以下问题,这看起来像是 PHP 7.1 和 7.2 中的错误。它与 &$this 引用和跨命名空间调用以及 call_user_func_arr
谁能解释一下下面“&”的作用: class TEST { } $abc =& new TEST(); 我知道这是引用。但是有人可以说明我为什么以及什么时候需要这样的东西吗?或者给我指向一个对此有很好解
引用变量是一个别名,也就是说,它是某个已存在变量的另一个名字。一旦把引用初始化为某个变量,就可以使用该引用名称或变量名称来指向变量。 C++ 引用 vs 指针 引用很容易与指针混淆,它们之间有三
目录 引言 背景 结论 引言 我选择写C++中的引用是因为我感觉大多数人误解了引用。而我之所以有这个感受是因为我主持过很多C++的面试,并且我很少
Perl 中的引用是指一个标量类型可以指向变量、数组、哈希表(也叫关联数组)甚至函数,可以应用在程序的任何地方 创建引用 定义变量的时候,在变量名前面加个 \,就得到了这个变量的一个引用 $sc
我编写了一个将从主脚本加载的 Perl 模块。该模块使用在主脚本中定义的子程序(我不是维护者)。 对于主脚本中的一个子例程,需要扩展,但我不想修补主脚本。相反,我想覆盖我的模块中的函数并保存对原始子例
我花了几个小时试图掌握 F# Quotations,但我遇到了一些障碍。我的要求是从可区分的联合类型中取出简单的函数(只是整数、+、-、/、*)并生成一个表达式树,最终将用于生成 C 代码。我知道使用
很多时候,问题(尤其是那些标记为 regex 的问题)询问验证密码的方法。似乎用户通常会寻求密码验证方法,包括确保密码包含特定字符、匹配特定模式和/或遵守最少字符数。这篇文章旨在帮助用户找到合适的密码
我想通过 MIN 函数内的地址(例如,C800)引用包含文本的最后一个单元格。你能帮忙吗? Sub Set_Formula() ' ----------------------------- Dim
使用常规的 for 循环,我可以做类似的事情: for (let i = 0; i < objects.length; i++) { delete objects[i]; } 常规的 for-
在 Cucumber 中,您定义了定义 BDD 语法的步骤;例如,您的测试可能有: When I navigate to step 3 然后你可以定义一个步骤: When /^I navigate t
这是什么UnaryExpression的目的,以及应该怎样使用? 最佳答案 它需要一个 Expression对象并用另一个 Expression 包裹它.例如,如果您有一个用于 lambda 的表达式
给出以下内容 $("#identifier div:first, #idetifier2").fadeOut(300,function() { // I need to reference jus
我不知道我要找的东西的正确术语,但我要找的是一个完整的引用,可以放在双引号之间的语句,比如 *, node()、@* 以及所有列出的 here加上任何其他存在的。 我链接到的答案提供了一些细节,但还
This question's answers are a community effort。编辑现有答案以改善此职位。它当前不接受新的答案或互动。 这是什么? 这是常见问答的集合。这也是一个社区Wi
Closed. This question does not meet Stack Overflow guidelines。它当前不接受答案。 想改善这个问题吗?更新问题,以便将其作为on-topic
考虑下一个代码: fn get_ref(slice: &'a Vec, f: fn(&'a Vec) -> R) -> R where R: 'a, { f(slice) } fn m
我是一名优秀的程序员,十分优秀!