一、基本技术

1)MATLAB索引或引用(MATLAB Indexing or Referencing)

在MATLAB中有三种基本方法可以选取一个矩阵的子阵。它们分别是下标法,线性法和逻辑法(subscripted, linear,

and

logical)。

1.1)下标法

非常简单,看几个例子就好。

A = 6:12;

A([3,5])

ans =

8 10

A([3:2:end])

ans =

8 10 12

A = [11 14 17;12 15 18;13 16

19];

A(2:3,2)

ans =

15

16

1.2)线性法

二维矩阵以列优先顺序可以线性展开,可以通过现行展开后的元素序号来访问元素。

A = [11 14 17;12 15 18;13 16

19];

A(6)

ans =

16

A([3,1,8])

ans =

13 11 18

A([3;1;8])

ans =

13

11

18

1.3)逻辑法

用一个和原矩阵具有相同尺寸的0-1矩阵,可以索引元素。在某个位置上为1表示选取元素,否则不选。得到的结果是一个向量。

A = 6:10;

A(logical([0 0 1 0 1]))

ans =

8 10

A =[1 2;3 4];

B = [1 0 0 1];

A(logical(B))

ans =

1 4

2)数组操作和矩阵操作(Array Operations vs. Matrix

Operations)

对矩阵的元素一个一个孤立进行的操作称作数组操作;而把矩阵视为一个整体进行的运算则成为矩阵操作。MATLAB运算符*,/,\,^都是矩阵运算,而相应的数组操作则是.*,

./, .\, .^

A=[1 0 ;0 1];

B=[0 1 ;1 0];

A*B % 矩阵乘法

ans =

0 1

1 0

A.*B % A和B对应项相乘

ans =

0 0

0 0

3)布朗数组操作(Boolean Array

Operations)

对矩阵的比较运算是数组操作,也就是说,是对每个元素孤立进行的因此其结果就不是一个“真”或者“假”,而是一堆“真假”。这个结果就是布朗数组。

D = [-0.2 1.0 1.5 3.0 -1.0 4.2

3.14];

D >= 0

ans =

0 1 1 1 0 1

1

如果想选出D中的正元素:

D = D(D>0)

D =

1.0000 1.5000 3.0000 4.2000 3.1400

除此之外,MATLAB运算中会出现NaN,Inf,-Inf。对它们的比较参见下例

Inf==Inf返回真

Inf<1返回假

NaN==NaN返回假

同时,可以用isinf,isnan判断,用法可以顾名思义。

在比较两个矩阵大小时,矩阵必须具有相同的尺寸,否则会报错。这是你用的上size和isequal,isequalwithequalnans(R13及以后)。

4)从向量构建矩阵(Constructing Matrices from

Vectors)

在MATLAB中创建常数矩阵非常简单,大家经常使用的是:

A =

ones(5,5)*10

但你是否知道,这个乘法是不必要的?

A = 10;

A = A(ones(5,5))

A =

10 10 10 10 10

10 10 10 10 10

10 10 10 10 10

10 10 10 10 10

10 10 10 10

10

类似的例子还有:

v = (1:5)';

n = 3;

M = v(:,ones(n,1))

M =

1 1 1

2 2 2

3 3 3

4 4 4

5 5 5

事实上,上述过程还有一种更加容易理解的实现方法:

A = repmat(10,[5 5]);

M = repmat([1:5]', [1,3]);

其中repmat的含义是把一个矩阵重复平铺,生成较大矩阵。

更多详细情况,参见函数repmat和meshgrid。

5)相关函数列表(Utility Functions)

ones 全1矩阵

zeros 全0矩阵

reshape 修改矩阵形状

repmat 矩阵平铺

meshgrid 3维plot需要用到的X-Y网格矩阵

ndgrid n维plot需要用到的X-Y-Z...网格矩阵

filter 一维数字滤波器,当数组元素前后相关时特别有用。

cumsum 数组元素的逐步累计

cumprod 数组元素的逐步累计

eye 单位矩阵

diag 生成对角矩阵或者求矩阵对角线

spdiags 稀疏对角矩阵

gallery 不同类型矩阵库

pascal Pascal

矩阵

hankel Hankel

矩阵

toeplitz Toeplitz

矩阵

二、扩充的例子

6)作用于两个向量的矩阵函数

假设我们要计算两个变量的函数F

F(x,y) = x*exp(-x^2 -

y^2)

我们有一系列x值,保存在x向量中,同时我们还有一系列y值。

我们要对向量x上的每个点和向量y上的每个点计算F值。换句话说,我们要计算对于给定向量x和y的所确定的网格上的F值。

使用meshgrid,我们可以复制x和y来建立合适的输入向量。然后可以使用第2节中的方法来计算这个函数。

x = (-2:.2:2);

y = (-1.5:.2:1.5)';

[X,Y] = meshgrid(x, y);

F = X .* exp(-X.^2 -

Y.^2);

如果函数F具有某些性质,你甚至可以不用meshgrid,比如

F(x,y) = x*y ,则可以直接用向量外积

x = (-2:2);

y = (-1.5:.5:1.5);

x'*y

在用两个向量建立矩阵时,在有些情况下,稀疏矩阵可以更加有效地利用存储空间,并实现有效的算法。我们将在第8节中以一个实例来进行更详细地讨论.

7)排序、设置和计数(Ordering, Setting, and Counting

Operations) 在迄今为止讨论过的例子中,对向量中一个元素的计算都是独立 于同一向量的其他元素的。但是,在许多应用中,你要做的计算 则可能与其它元素密切相关。例如,假设你用一个向量x来表示一 个集合。不观察向量的其他元素,你并不知道某个元素是不是一 个冗余元素,并应该被去掉。如何在不使用循环语句的情况下删除 冗余元素,至少在现在,并不是一个明显可以解决的问题。

解决这类问题需要相当的智巧。以下介绍一些可用的基本工具

max 最大元素

min 最小元素

sort 递增排序

unique 寻找集合中互异元素(去掉相同元素)

diff 差分运算符[X(2) - X(1), X(3) - X(2), ... X(n) -

X(n-1)]

find 查找非零、非NaN元素的索引值

union 集合并

intersect 集合交

setdiff 集合差

setxor 集合异或

继续我们的实例,消除向量中的多余元素。注意:一旦向量排序后, 任何多余的元素就是相邻的了。同时,在任何相等的相邻元素在向量 diff运算时变为零。这是我们能够应用以下策略达到目的。我们现在 在已排序向量中,选取那些差分非零的元素。

% 初次尝试。不太正确!

x = sort(x(:));

difference = diff(x);

y = x(difference~=0);

这离正确结果很近了,但是我们忘了diff函数返回向量的元素个数比 输入向量少1。在我们的初次尝试中,没有考虑到最后一个元素也可能 是相异的。为了解决这个问题,我们可以在进行差分之前给向量x加入 一个元素,并且使得它与以前的元素一定不同。一种实现的方法是增 加一个NaN。

% 最终的版本。

x = sort(x(:));

difference = diff([x;NaN]);

y =

x(difference~=0);

我们使用(:)运算来保证x是一个向量。我们使用~=0运算,而不用find 函数,因为find函数不返回NaN元素的索引值,而我们操作中差分的最 后元素一定是NaN。这一实例还有另一种实现方式:

y=unique(x);

后者当然很简单,但是前者作为一个练习并非无用,它是为了练习使用 矢量化技术,并示范如何编写你自己的高效代码。此外,前者还有一个 作用:Unique函数提供了一些超出我们要求的额外功能,这可能降低代 码的执行速度。

假设我们不只是要返回集合x,而且要知道在原始的矩阵里每个相异元素 出现了多少个“复本”。一旦我们对x排序并进行了差分,我们可以用 find来确定差分变化的位置。再将这个变化位置进行差分,就可以得到 复本的数目。这就是"diff

of find of

diff"的技巧。基于以上的讨论, 我们有:

% Find the redundancy in a vector

x

x = sort(x(:));

difference =

diff([x;max(x)+1]);

count =

diff(find([1;difference]));

y = x(find(difference));

plot(y,count)

这个图画出了x中每个相异元素出现的复本数。注意,在这里我们避开了 NaN,因为find不返回NaN元素的索引值。但是,作为特例,NaN和Inf 的复本数可以容易地计算出来:

count_nans =

sum(isnan(x(:)));

count_infs =

sum(isinf(x(:)));

另一个用于求和或者计数运算的矢量化技巧是用类似建立稀疏矩阵的方 法实现的。这还将在第9节中作更加详细的讨论.

8)稀疏矩阵结构(Sparse Matrix

Structures)

在某些情况下,你可以使用稀疏矩阵来增加计算的效率。如果你构造一 个大的中间矩阵,通常矢量化更加容易。在某些情况下,你可以充分利 用稀疏矩阵结构来矢量化代码,而对于这个中间矩阵不需要大的存储空 间。

假设在上一个例子中,你事先知道集合y的域是整数的子集,

{k+1,k+2,...k+n};即,

y = (1:n) +

k

例如,这样的数据可能代表一个调色板的索引值。然后,你就可以对集 合中每个元素的出现进行计数(构建色彩直方图?译者)。这是对上一 节中"diff

of find of diff"技巧的一种变形。

现在让我们来构造一个大的m x

n矩阵A,这里m是原始x向量中的元素数, n是集合y中的元素数。

A(i,j) = 1 if x(i) =

y(j)

0 otherwise

回想一下第3节和第4节,你可能认为我们需要从x和y来构造矩阵A。如果 当然可以,但要消耗许多存储空间。我们可以做得更好,因为我们知道, 矩阵A中的多数元素为0,x中的每个元素对应的行上只有一个值为1。

以下就是构造矩阵的方法(注意到y(j) =

k+j,根据以上的公式):

x = sort(x(:));

A = sparse(1:length(x), x+k, 1, length(x),

n);

现在我们对A的列进行求和,得到出现次数。

count =

sum(A);

在这种情况下,我们不必明确地形成排序向量y,因为我们事先知道

y = 1:n + k.

这里的关键是使用数据,(也就是说,用x控制矩阵A的结构)。由于x在 一个已知范围内取整数值,我们可以更加有效地构造矩阵。

假设你要给一个很大矩阵的每一列乘以相同的向量。使用稀疏矩阵,不仅 可以节省空间,并且要比在第5节介绍的方法更加快速.

下面是它的工作 方式:

F = rand(1024,1024);

x = rand(1024,1);

% 对F的所有行进行点型乘法.

Y = F * diag(sparse(x));

% 对F的所有列进行点型乘法.

Y = diag(sparse(x)) *

F;

我们充分利用矩阵乘法算符来执行大规模运算,并使用稀疏矩阵以防止临 时变量变得太大。

9)附加的例子(Additional

Examples)

下面的例子使用一些在本技术手册中讨论过的方法,以及其它一些相关方 法。请尝试使用tic

和toc(或t=cputime和cputime-t),看一下速度加快 的效果。

用于计算数组的双重for循环。

使用的工具:数组乘法

优化前:

A = magic(100);

B = pascal(100);

for j =

1:100

for k =

1:100;

X(j,k) = sqrt(A(j,k)) * (B(j,k) -

1);

end

end

优化后:

A = magic(100);

B = pascal(100);

X = sqrt(A).*(B-1);

用一个循环建立一个向量,其元素依赖于前一个元素

使用的工具:FILTER, CUMSUM,

CUMPROD

优化前:

A = 1;

L = 1000;

for i = 1:L

A(i+1) = 2*A(i)+1;

end

优化后:

L = 1000;

A = filter([1],[1

-2],ones(1,L+1));

如果你的向量构造只使用加法或乘法,你可使用cumsum或cumprod函数。

优化前:

n=10000;

V_B=100*ones(1,n);

V_B2=100*ones(1,n);

ScaleFactor=rand(1,n-1);

for i =

2:n

V_B(i) =

V_B(i-1)*(1+ScaleFactor(i-1));

end

for

i=2:n

V_B2(i) = V_B2(i-1)+3;

end

优化后:

n=10000;

V_A=100*ones(1,n);

V_A2 = 100*ones(1,n);

ScaleFactor=rand(1,n-1);

V_A=cumprod([100

1+ScaleFactor]);

V_A2=cumsum([100

3*ones(1,n-1)]);

向量累加,每5个元素进行一次:

工具:CUMSUM , 向量索引

优化前:

% Use an arbitrary vector,

x

x = 1:10000;

y = [];

for n =

5:5:length(x)

y = [y sum(x(1:n))];

end

优化后(使用预分配):

x = 1:10000;

ylength = (length(x) -

mod(length(x),5))/5;

% Avoid using ZEROS command during

preallocation

y(1:ylength) = 0;

for n =

5:5:length(x)

y(n/5) = sum(x(1:n));

end

优化后(使用矢量化,不再需要预分配):

x = 1:10000;

cums = cumsum(x);

y = cums(5:5:length(x));

操作一个向量,当某个元素的后继元素为0时,重复这个元素:

工具:FIND, CUMSUM,

DIFF

任务:我们要操作一个由非零数值和零组成的向量,要求把零替换成为 它前面的非零数值。例如,我们要转换下面的向量:

a=2; b=3; c=5; d=15; e=11;

x = [a 0 0 0 b 0 0 c 0 0 0 0 d 0 e 0 0 0 0

0];

为:

x = [a a a a b b b c c c c c d d e e e e e

e];

解(diff和cumsum是反函数):

valind = find(x);

x(valind(2:end)) =

diff(x(valind));

x =

cumsum(x);

将向量的元素累加到特定位置上

工具:SPARSE

优化前:

% The values we are summing at designated

indices

values = [20 15 45 50 75 10 15 15 35 40

10];

% The indices associated with the values are summed

cumulatively

indices = [2 4 4 1 3 4 2 1 3 3

1];

totals =

zeros(max(indices),1);

for n =

1:length(indices)

totals(indices(n)) = totals(indices(n)) +

values(n);

end

优化后:

indices = [2 4 4 1 3 4 2 1 3 3

1];

totals =

full(sparse(indices,1,values));

注意:这一方法开辟了稀疏矩阵的新用途。在使用sparse命令创建稀疏矩阵 时,它是对分配到同一个索引的所有值求和,而不是替代已有的数值。这称 为"向量累加",是MATLAB处理稀疏矩阵的方式。

matlab 矢量化,matlab矢量化编程简要相关推荐

  1. GPS数据矢量化JAVA_SVGX矢量化图形编辑器,100%JAVA实现的矢量化图形编辑器

    SVGX矢量化图形编辑器,100%JAVA实现的矢量化图形编辑器 SVGX矢量化图形编辑器是面向工程应用的矢量图形制作软件,基于著名的Eclipse GEF图形编辑框架实现了W3C SVG 1.1规范 ...

  2. matlab图片矢量化,matlab图形矢量化解决方案

    大致思路:matlab中生成矢量格式文件-导入Visio中-编辑-导出合适格式-在其他软件中使用 准备工具 Matlab 2014b或更高版本 Visio 2007或更高版本 我查看过,Matlab能 ...

  3. matlab 矢量化编程(二)—— 使用 meshgrid

    matlab 矩阵矢量化编程 使用 meshgrid 使用 meshgrid 避免二重循环. patchSize = 17;pixel_weights = zeros(patchSize); mid ...

  4. Stanford UFLDL教程 矢量化编程

    矢量化编程 当使用学习算法时,一段更快的代码通常意味着项目进展更快.例如,如果你的学习算法需要花费20分钟运行完成,这意味着你每个小时能"尝试"3个新主意.但是假如你的程序需要20 ...

  5. matlab 矢量化编程(四)—— 标量函数转化为能够处理矢量的函数

    1. 组合的矢量实现 nchoosek(n, k) 的第二个参数在 matlab 下是不支持矢量化的,必须是标量形式.但 matlab 下的 gamma 函数,却可支持,矢量形式,又因为,gamma ...

  6. matlab 矩阵矢量化编程

    如我们想验证: ∑nxnxTn=XXT \sum_nx_nx_n^T=XX^T 其中 xn,n=1,-,Nx_n,n=1,\ldots,N分别表示 XX的每一列 % 循环的做法 T = zeros(s ...

  7. EPI_H/EPI_V(边缘保持指数,matlab 矢量化编程)

    EPI: edge preservation index,衡量对原始图像的操作(目标图像)对图像边缘的保持能力. EPI_H:horizontal ,水平方向: EPI_V:vertical,垂直方向 ...

  8. matlab 矢量化编程(三) —— 软阈值函数

    dj,k^=⎧⎩⎨⎪⎪dj,k−λ,dj,k≥λ0,otherwisedj,k+λ,dj,k≤−λ \hat{d_{j,k}}=\left\{\begin{array}{l}d_{j,k}-\lamb ...

  9. MATLAB—数组运算及数组化编程

    文章目录 前言 一.数组的结构和创建 1.数组及其结构 2.行数组的创建 3.对数组构造的操作 二.数组元素编址及寻访 1.数组元素的编址 2.二维数组元素的寻访 三.数组运算 非数的问题 前言 编程 ...

最新文章

  1. python中数据分析的流程为-用Python进行数据分析-1
  2. X264代码中一些参数的意义
  3. Nginx突破高并发的性能优化 - 运维笔记
  4. Taro+react开发(28)小程序怎么进行自适应
  5. vgc机器人编程1到13题_工业机器人编程与实操-期末试题
  6. 在线网页快捷方式创建工具
  7. OZ Report 오즈 리포트 개발
  8. 浪潮n系列服务器指示灯_中国服务器市场,浪潮跑出,联想和华为出现衰退
  9. 使用 Laravel 5.5+ 更好的来实现 404 响应
  10. python贪心算法几个经典例子_Python笔试——贪心算法
  11. 【资源】1497- Vue超全资源,收藏!
  12. git使用时报错:fatal: unable to access ‘xxx‘ : Failed to connect to github.com port 443 after: 【Time out】
  13. kotlin版贪吃蛇小游戏
  14. 汇编中esp和ebp在函数栈空间的保存和变化 call的参数和局部变量的关系详解
  15. (CSP2019·J T4)加工零件【spfa】【最短路】
  16. c语言十全十美游戏规则,十全十美游戏
  17. html5支持4k视频播放器,哪个是最好的4K视频播放器?五个最佳播放软件(个人经验)...
  18. AHB-SRAM简单设计之内部模块 sram_core.v
  19. 隐藏服务器端信息X-Powered-By: Servlet/3.0
  20. Python 数字黑洞

热门文章

  1. C# 全角半角相互转换
  2. 简述List、Set、Map类型的集合的各自特点
  3. java 实现超时_如何实现带有超时的Runnable? - java
  4. 字段 新增hive_Hive分区表 | 每日五分钟学大数据
  5. express+vue+mongodb+session 实现注册登录
  6. Thread类中的join方法
  7. css3 - target
  8. 手机百度输入法的用户体验
  9. [2-sat]HDOJ3062 Party
  10. 45个极具冲击力的WordPress摄影网站模板