0

我正在尝试对图像中的每个像素进行如下操作:

方程

A 是 x*y*4 矩阵,w 只是一个 1x9 向量。

我有一个矩阵 L,它是 200x200x4x9。

L 的前两个维度是像素的 (x,y) 位置。每个位置有 4 个不同的子像素(第三维)。每个子像素都有一个向量 m,最后一个维度是我方程中的 m。

我计划获得整个图像的第一个子像素的结果,这就是我尝试过的:

A (:,:,1) = w * L (:, :, 1, :)  ====> Inputs must be 2-D, or at least one input must be scalar.
A (:,:,1) = w * L (:, :, 1, :)' ====> Transpose on ND array is not defined.
A (:,:,1) = w * reshape (L (:, :, 1, :), 1, 9)' ===> To RESHAPE the number of elements must not change.

如果我只打印 L (1,1,1,:) 我会得到单个元素的值(看起来不像向量):

ans(:,:,1,1) = 0.8980
ans(:,:,1,2) = 0.8065
ans(:,:,1,3) = 0.8471
ans(:,:,1,4) = 0.7607
ans(:,:,1,5) = 0.7175
ans(:,:,1,6) = 0.9020
ans(:,:,1,7) = 0.8100
ans(:,:,1,8) = 0.7640
ans(:,:,1,9) = 0.8135

编辑:供参考,

Size(A) = [200 200 4]
Size(L) = [200 200 4 9] 
Size(w) = [1 9]

编辑:这就是我使用循环的方式

环形

4

4 回答 4

2

与其在双 for 循环中使用带有 reshape 等的矩阵乘法,不如在更短的维度上反转过程并循环,如下所示:

A=zeros(size(L)(1:3));
for i=1:9
    A(:,:,:)=A(:,:,:)+w(i).*L(:,:,:,i);
end

结果,不是通过双 for 循环进行 40000 次循环,而是总共进行了 9 次循环。而且我怀疑 Matlab 的 JITA 也会在这样的循环上工作,以产生更好的结果(我使用八度音阶,所以我不能确定 - 这也不值得我做任何时序测试,因为八度音阶的时序不会对matlab有效)。

也可以通过简单地使用向量索引而不是矩阵索引来纯粹以向量形式进行。它是这样工作的:

A=zeros(size(L)(1:3));
A(:)=reshape(L,numel(A),9)*w';

这个版本当然使用了 reshape,但它避免了 bsxfun 和挤压,我怀疑它会比其他解决方案更快,除了前面提到的 for 循环超过 9 个值。

(当然,如果要将其限制为仅“第一个子像素”,则需要限制数据。对于 for 循环选项,只需在适当的位置放置一个 1 代替冒号即可。对于重塑和矩阵相乘选项,在它出现的两个地方用 L(:,:,1,:) 替换 L)

于 2013-03-17T03:23:39.493 回答
1

我最近成为了它的忠实粉丝bsxfun,假设我对您的理解正确,这似乎非常适合它。我们只需要重塑w然后利用bsxfun' 自动扩展单例维度的能力来使其输入的大小匹配。

w = reshape(w, [1, 1, 1, length(w)]);
A = sum(bsxfun(@times, L, w), 4);
于 2013-03-15T19:17:23.987 回答
1

您应该使用squeeze来获得矢量形式。尝试

a=squeeze(L(1,1,1,:))
于 2013-03-15T16:15:37.013 回答
0

多维乘法在 Matlab 中不起作用,这就是为什么你不能这样做A = L * w

从 Mathworks 页面(如何在 MATLAB 中执行多维矩阵乘法?):

解决方案:

在 MATLAB 中执行多维矩阵乘法的功能不可用。

作为一种解决方法,请使用 FOR 循环。

此外,还有一个名为 NDFUN 的用户创建函数,用于执行 ND 矩阵乘法。有关详细信息,请参阅以下 URL 中有关 NDFUN 的部分:

http://www.mit.edu/~pwb/matlab/

带有循环的解决方案应如下所示:

A = zeros( size(L)(1:3) );
for i = 1:size(L,1)
    for j = 1 : size(L,2)
        A(i,j,:) = squeeze( L(i,j,:,:) ) * w';
    end
end

但是,如果您更喜欢简洁的解决方案而不是更快的解决方案(循环应该很快),您可以使用答案中的方法将 3D 矩阵与 2D 矩阵相乘(假设size(L) == [200 200 4 9]size(w) = [1 9]):

Ac = cellfun( @(x) squeeze(x)' * w', num2cell(L,4), 'UniformOutput', false);
A = cell2mat( squeeze(Ac) );
于 2013-03-15T16:24:33.973 回答