如何在MATLAB中可视化3D矩阵

时间:2012-06-15 19:10:45

标签: matlab memory-management matrix plot visualization

我正在尝试在MATLAB中可视化包含3D数组的数据。

阵列的尺寸为20 * 20 * 40,除了部分元素外都是零。

我正在寻找一种在散点图中绘制这些非零点的方法,以便非零点相互连接。

这是我到目前为止所做的:

b=zeros(6,6,3);
a=[1 1 1;2 2 1;2 1 1;1 2 1;
1 1 2;1 2 2;2 1 2;2 2 2;
6 6 3;6 5 3;5 6 3;5 5 3;
6 6 2;6 5 2;5 6 2;5 5 2];
[r c]=size(a);
for i = 1:r
 b(a(i,c-2),a(i,c-1),a(i,c)) = 1;
end
[m n o]=size(b);
figure (1)
[x,y,z] = meshgrid(1:m,1:n,1:o);
scatter3(x(:),y(:),z(:),90,b(:),'filled')

所以,我所追求的是能够将八个点中的每一个连接到立方晶格。任何想法都非常感谢。 enter image description here

修改 非常感谢所有帮助过很多的专家。现在,我面临另一个记忆问题。

我的实例的b矩阵是1000 * 1000 * 2000,我有一个大小为4,4700,000 * 3的“a”矩阵。 “a”矩阵的所有元素都是整数值。 虽然我有高达48GB的内存可用性。但是,在程序上方的“for循环”中,程序会弹回“内存不足”错误。

非常感谢任何提高内存效率的想法。

3 个答案:

答案 0 :(得分:3)

考虑以下内容(基于previous answer):

%# adjacency matrix
adj = false(numel(b));

%# extract first cube, and connect all its points
bb = b;
bb(:,1:3,:) = false;
idx = find(bb);
adj(idx,idx) = true;

%# extract second cube, and connect all its points
bb = b;
bb(:,4:6,:) = false;
idx = find(bb);
adj(idx,idx) = true;

%# points indices
[r c] = find(adj);
p = [r c]';

%# plot
plot3(x(p), y(p), z(p), 'LineWidth',2, 'Color',[.4 .4 1], ...
    'Marker','o', 'MarkerSize',6, ...
    'MarkerFaceColor','g', 'MarkerEdgeColor','g')
axis equal, axis vis3d, grid on, view(3)
xlabel x, ylabel y, zlabel z

enter image description here


编辑:

手动连接点会更容易:

%# edges: connecting points indices
p = [
    1 2; 2 8; 8 7; 7 1;
    37 38; 38 44; 44 43; 43 37;
    1 37; 2 38; 7 43; 8 44;
    65 66; 66 72; 72 71; 71 65;
    101 102; 102 108; 108 107; 107 101;
    65 101; 66 102; 71 107; 72 108
]';

%# plot
figure
plot3(x(p), y(p), z(p), 'LineWidth',2, 'Color',[.4 .4 1], ...
    'Marker','o', 'MarkerSize',6, ...
    'MarkerFaceColor','g', 'MarkerEdgeColor','g')
axis equal, axis vis3d, grid on, view(3)
xlabel x, ylabel y, zlabel z

%# label points
labels = strtrim(cellstr( num2str((1:numel(b))','%d') ));
idx = find(b);
text(x(idx(:)), y(idx(:)), z(idx(:)), labels(idx(:)), ...
    'Color','m', ...
    'VerticalAlignment','bottom', 'HorizontalAlignment','left')

screenshot


编辑#2 :(功劳归于@tmpearce)

您可以半自动化构建边列表的过程。只需使用以下代码替换上面代码中手动构造的matrx p

%# compute edges: pairs of vertex indices 
yIdx = {1:3 4:6};           %# hack to separate each cube points
p = cell(numel(yIdx),1);
for i=1:numel(yIdx)         %# for each cube
    %# find indices of vertices in this cube
    bb = b;
    bb(:,yIdx{i},:) = false;
    idx = find(bb);

    %# compute L1-distance between all pairs of vertices,
    %# and find pairs which are unit length apart
    [r,c] = find(triu(squareform(pdist([x(idx) y(idx) z(idx)],'cityblock'))==1));

    %# store the edges
    p{i} = [idx(r) idx(c)]';
end
p = cat(2,p{:});            %# merge all edges found

我们的想法是,对于每个立方体,我们计算所有顶点之间的city-block distance。 “side”边缘的距离为1,而对角线的距离大于或等于2

此过程仍假设属于每个多维数据集的顶点列表已提供给我们,或者像我们在上面的代码中那样轻松提取...

答案 1 :(得分:2)

我想我会回答Amro答案的后续行动,并展示如何自动连接沿立方矩阵的边缘。这个答案旨在成为Amro的补充,我建议将其纳入该解决方案,以便将来更容易阅读。

这从Amro的解决方案中的idx开始,该解决方案包含多维数据集角落的线性索引。下面我只用了一个立方体,所以矩阵不是很大,我们可以在这里看到它们。

>> idx' %# transposed to save space in the printed output
ans =
 1     2     7     8    37    38    43    44 '

%# now get the subindex of each point in 3D
>> [sx sy sz] = ind2sub([6 6 3],idx);
%# calculate the "edge distance" between each corner (not euclidean distance)
>> dist = abs(bsxfun(@minus,sx,sx')) +...
          abs(bsxfun(@minus,sy,sy')) +...
          abs(bsxfun(@minus,sz,sz'))
dist =

 0     1     1     2     1     2     2     3
 1     0     2     1     2     1     3     2
 1     2     0     1     2     3     1     2
 2     1     1     0     3     2     2     1
 1     2     2     3     0     1     1     2
 2     1     3     2     1     0     2     1
 2     3     1     2     1     2     0     1
 3     2     2     1     2     1     1     0 '

%# find row and column index of points one edge apart
>> [r c]=find(triu(dist==1)); %# triu means don't duplicate edges

%# create 'p' matrix like in Amro's answer
>> p = [idx(r) idx(c)]'
p =
 1     1     2     7     1     2    37     7    37     8    38    43
 2     7     8     8    37    38    38    43    43    44    44    44

为了形象化,你可以完全像Amro的回答一样。

答案 2 :(得分:0)

这是你需要的吗?

b=logical(b);                                  
scatter3(x(b(:)),y(b(:)),z(b(:)),90,'filled')

enter image description here