gpt4 book ai didi

c++ - 使用 Eigen::Map 作为 Eigen::MatrixXd 类型的函数参数

转载 作者:塔克拉玛干 更新时间:2023-11-03 02:01:51 27 4
gpt4 key购买 nike

简而言之,问题是如何通过一个

Eigen::Map<Eigen::MatrixXd>

反对一个期待一个函数

Eigen::MatrixXd

对象。


长话短说:

我有这个 C++ 函数声明

void npMatrix(const Eigen::MatrixXd &data, Eigen::MatrixXd &result);

连同这个实现

void npMatrix(const Eigen::MatrixXd &data, Eigen::MatrixXd &result)
{
//Just do s.th. with arguments
std::cout << data << std::endl;

result(1,1) = -5;
std::cout << result << std::endl;
}

我想使用 numpy.array 作为参数从 python 调用这个函数。为此,我使用了一个用 c++ 编写的包装器函数

void pyMatrix(const double* p_data, const int dimData[],
double* p_result, const int dimResult[]);

它接受一个指向数据的指针、数据数组的大小、一个指向结果的指针和结果数组的大小。数据指针指向内存的 const 补丁,因为当为结果保留的内存补丁是可写的时,数据不会被更改。函数的实现

void pyMatrix(const double *p_data, const int dimData[], double *p_result, const int dimResult[])
{
Eigen::Map<const Eigen::MatrixXd> dataMap(p_data, dimData[0], dimData[1]);
Eigen::Map<Eigen::MatrixXd> resultMap(p_result, dimResult[0], dimResult[1]);

resultMap(0,0) = 100;

npMatrix(dataMap, resultMap);
}

分别为数据和结果定义一个 Eigen::Map。 Eigen::Map 允许作为一种 Eigen::Matrix 访问原始内存。 dataMap是类型

<const Eigen::MatrixXd>

因为关联的内存是只读的; resultMap 相比之下是类型

<Eigen::MatrixXd>

因为它必须是可写的。线路

resultMap(0,0) = 100;

表明,resultMap 实际上是可写的。在将 dataMap 传递给预期 const Eigen::MatrixXd 起作用的 npMatrix() 时,我找不到以相同方式传递 resultMap 的方法。我敢肯定,麻烦来自于 npMatrix 的第一个参数是 const,而第二个不是。我找到的一个可能的解决方案是定义

Eigen::MatrixXd resultMatrix = resultMap;

并将此 resutlMatrix 传递给 npMatrix()。但是,我猜,这会创建一个拷贝,因此会破坏 Eigen::Map 的良好内存映射。所以我的问题是。

有没有办法将 Eigen:Map 传递给需要非常量 Eigen::MatrixXd 的函数?

附带说明:我可以将 npMatrix 更改为期望 Eigen::Map,但由于在实际项目中,函数已经存在并经过测试,我宁愿不对它们进行调整。

为了完成这个问题,这里是调用 pyMatrix() 的 python 文件

import ctypes as ct
import numpy as np
import matplotlib.pyplot as plt

# Load libfit and define input types
ct.cdll.LoadLibrary("/home/wmader/Methods/fdmb-refactor/build/pyinterface/libpyfit.so")
libfit = ct.CDLL("libpyfit.so")

libfit.pyMatrix.argtypes = [np.ctypeslib.ndpointer(dtype=np.float64, ndim=2),
np.ctypeslib.ndpointer(dtype=np.int32, ndim=1),
np.ctypeslib.ndpointer(dtype=np.float64, ndim=2, flags='WRITEABLE'),
np.ctypeslib.ndpointer(dtype=np.int32, ndim=1)
]

data = np.array(np.random.randn(10, 2), dtype=np.float64, order='F')
result = np.zeros_like(data, dtype=np.float64, order='F')

libfit.pyMatrix(data, np.array(data.shape, dtype=np.int32),
result, np.array(result.shape, dtype=np.int32))

最佳答案

将其作为指向数据的普通指针传递,然后 Eigen::Map 将其映射到那里。或者,使用 template <typename Derived>等等,在 http://eigen.tuxfamily.org/dox/TopicFunctionTakingEigenTypes.html 中找到不过,我个人的选择是第一个,因为最好是拥有不暴露您使用的每个 API 的所有顽固性的代码。此外,您不会失去与 eigen 的兼容性,也不会与您(或其他任何人)以后可能使用的任何其他类型的库失去兼容性。

我还发现了另一个技巧,可以在很多场合使用:

Eigen::MatrixXd a;
//lets assume a data pointer like double* DATA that we want to map
//Now we do
new (&a) Eigen::Map<Eigen::Matrix<Double,Eigen::Dynamic,Eigen::Dynamic>> (DATA,DATA rows,DATA cols);

这将按照您的要求进行操作,而不会浪费内存。我认为这是一个很酷的技巧,a将表现为 matrixXd,但我没有测试过每一个场合。 它没有内存拷贝。但是,您可能需要调整大小 a在分配之前调整到合适的大小。即便如此,编译器不会在您请求 resize 时立即分配所有内存。操作,所以也不会有大量无用的内存分配!

Be careful! Resizing operations might reallocate the memory used by an eigen matrix! So, if you ::Map a memory but then you perform an action that resizes the matrix, it might be mapped to a different place in memory.

关于c++ - 使用 Eigen::Map<Eigen::MatrixXd> 作为 Eigen::MatrixXd 类型的函数参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28587893/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com