利用矩阵的索引取出原矩阵的子集元素是一种有效的方式。MATLAB的多种索引不但类型强大、灵活,而且表达清晰易读。在理解电脑MATLAB编程方面。体会MATLAB以矩阵为导向思想的高效性,掌握索引便是一种最佳方式。
索引也和MATLAB用户经常听到的另一个属于“矢量/向量”紧密相关。矢量化意味着使用MATLAB的语法结构替代循环这一语法,能够使程序运行得更快、更具有可读性。当今大多数向量化的技术,许多都是借鉴了MATLAB的索引思想。在这篇文章中,将要学习其中的五种。
如果需要学习更多其他类似的方法,敬请参阅本文章结尾处的所列资源。
索引向量
让我们以一个简单向量为例,该向量仅有一个下标,向量:
v = [ 16 5 9 4 2 11 7 14];
下标可以是一个单一的值.
v (3) %取出第三个元素
ans =
9
或者下标本身可以是一个向量:
v( [1 5 6]) %取出第1、5、6个元素
ans =
16 2 11
MATLAB的冒号“:”提供了一个简单方式取出向量v中连续范围的元素
v (3 : 7) % 取出从第3到第7个的所有元素
ans =
9 4 2 11 7
向量V的两个分区可以组成一个新的新的向量:
v([5:8 1:4])
ans =
1 至 7 列
2 11 7 14 16 5 9
8 列
4
特殊符号”end"快速取出向量v中最后一个元素:
v(end) % 取出最后一个元素
ans =
14
v(5:end) % 取出从第5个到最后一个元素
ans =
2 11 7 14
你甚至可以对符号”end“进行数学运算:
v(2:end-1) % 取出从第2个到向量倒数第2个的所有元素
ans =
5 9 4 2 11 7
将冒号":'和”end"组合起来可以产生巨大效果,例如取出一段范围内以k为步进的向量元素集
v(1:2:end) % 取出奇数下标的元素
ans =
16 9 2 7
v(end:-1:1) % 逆序取出所有元素
ans =
1 至 7 列
14 7 11 2 4 9 5
8 列
16
通过在等式左边使用索引表达式,你可以替换向量中特定元素:
v([2 3 4]) = [10 15 20]% 替代v中一些特定元素
v = 16 10 15 20 2 11 7 14
等式右边的元素必须和等式左边下标对应的元素保持一致。然而,你可以在等式右边用标量
v([2 3]) = 30 % 第2和第3元素重新赋值为30
v =
1 至 7 列
16 30 30 4 2 11 7
8 列
14
这种索引赋值称作 标量扩张
带有两个下标的索引矩阵
现在考虑矩阵索引的用法,我们用一个魔力方阵来做实验:
A = magic(4)
A=
16 2 3 13
5 11 10 8
9 7 6 12
4 14 15 1
通常,矩阵索引用两个下标 --一个作为行下标,一个作为列下标,最简单方法取出矩阵一个元素如下:
A(2,4) % 取出第2行,第4列对应元素
ans =
8
更为常用的是,一个行下标或列下标可以是向量:
A(2:4,1:2)
ans =
5 11
9 7
4 14
在下标位置的一个符号":",是对”1:end“方便快捷的替代,总是用来选整行或整列元素:
A(3, :) % 取出完整的第3行
ans =
9 7 6 12
A(:,end) % 取出完整的最后一列
ans =
13
8
12
1
在这儿学习者总是迷惑:如何从矩阵中去取分散的几个元素?例如,假定你想从矩阵A中取出坐标为(2,1),(3,2), 和 (4,4)的元素,表达式
A([2 3 4], [1 2 4])取出的显然不是你所想要的,下面([2 4], [1 2 4])是如何工作的:
A([2 3 4], [1 2 4])
ans =
5 11 8
9 7 12
4 14 1
线性索引
表达式A(14)表示什么呢?
当你用带一个下标的索引取出矩阵元素时,MATLAB默认矩阵中所有元素一个长列向量
来存储数据,如下:
16
5
9
...
8
12
1
因此,表达式A(14)取出列向量的第14个元素,用单一下标取出数据元素的方式称作:线性索引
你可以知道A(14)和A(2,4)等价
单一下标可以是一个多个线性索引组合而成的向量,如下:
A([6 12 15])
ans =
11 15 12
再次考虑这个问题:从矩阵A中提取(2,1),(3,2), 和(4,4) 对应的元素?你可以通过用线性
索引来取出这些元素:
A([2 7 16])
ans =
5 7 1
从这个例子中你可以很简单看出索引,但是长远的你如何计算线性索引呢?MATLAB提供一个函数:sub2ind,该函数能够将特定行列下标转换为线性索引。你可以用这种方式提取特定元素:
idx = sub2ind(size(A), [2 3 4], [1 2 4])
ans =
2 7 16
A(idx)
ans =
5 7 1
关于用线性索引的最新例子:
例1:平移一个矩阵的数排
一个MATLAB用户最近在comp.soft-sys.matlab新手组提出一个问题:
如果我想移动矩阵m-n列k次,我用A(:,[n-k+1:n1:n-k])实现,当如果k是一个数字集怎么办?也就是说,倘若k是一个长度为m的向量呢,是否有快捷方便的办法做到呢?
社区新手组贡献者,Peter Acklam, 提出了用”sub2ind“和线性索引来解决:
%这个例子我没看,以后有工夫再研究
% index vectors for rows andcolumns
p = 1:m;
q = 1:n;
% index matrices for rows andcolumns
[P, Q] = ndgrid(p, q);
% create a matrix with the shiftvalues
K = repmat(k(:), [1 n]);
% update the matrix with the columnindexes
Q = 1 + mod(Q+K-1, n);
% create matrix of linear indexes
ind = sub2ind([m n], P, Q);
% finally, create the output matrix
B = A(ind);
例2:使矩阵的一些元素为0
另一个MATLAB用户提出这个问题。
我想获取每一个的最大值,这不是什么大问题,但是之后我想令其他元素为0,比如,矩阵:
1 2 3 4
5 5 6 5
7 9 8 3
should become:
0 0 0 4
0 0 6 0
0 9 0 0
另一个社区贡献者,BrettShoelson,提出了解决办法:
%这里用到的函数很简单,搜一下就知道用法啦
[Y,I]= max(A, [], 2);%获取每一行最大值,结果 存在列向量Y中,I里存的是每行最大值的列位置。
B = zeros(size(A));%建立一个和A同等大小的矩阵
B(sub2ind(size(A), 1:length(I),I')) = Y;%将三个最大值分别放到线性索引对应的位置上
逻辑索引
另一个索引变量,逻辑索引,已被证明其有用并且易于表达。在逻辑索引中,你可以为矩阵下标使用一个单一的逻辑数组,MATLAB取出逻辑判断为真的元素,输出经常是一个列向量。例如,A(A>2)取出矩阵A中所有大于12的元素
A(A > 12)
ans =
16
14
15
13
许多MATLAB函数以“is"开头,返回逻辑数组,这对于逻辑索引非常有用。
例如,你可以用另外一个值来替代一个数组中所有不是数字的数:NaNs,这里使用函数isnan,逻辑索引和标量扩张。矩阵B中所有NaN元素都赋值为0
B(isnan(B)) = 0
或者亦可以将字符串矩阵中所用空格用下划线‘_'代替
str(isspace(str)) = '_'
逻辑索引和函数find关系紧密。表达式A(A>5)等价于A(find(A>5))。你所用哪种主要取决于你的风格和代码可读性
但是这种取舍也取决于是否你需要实际的索引去做一些其他的计算。
例如,假定你想暂时将矩阵中NaN(不为数字的元素)赋值为0,做一些运算。之后将NaN(不为数字的元素)重新放回矩阵原位置。在这个例子中,2维滤除使用函数:filter2,你可能会这样做:
nan_locations = find(isnan(A));
A(nan_locations) = 0;
A = filter2(ones(3,3), A);
A(nan_locations) = NaN;
我们希望在这一篇文章中列举的MATLAB索引变量能够给你一个感觉,你可以高效精简的描述算法。在你的MATLAB编程中,使用这些技术和与之相关的函数能够增强你创造出熟练地、易读地和矢量化的代码的能力.
参考文献
王茂春于2015年12月27日 东六实验室
MATLAB中的矩阵索引