002_avoid_for_loop_in_Matlab避免使用for循环

避免使用for循环

在程序设计思想中,循环是一个很有力的工具。在循环中,计算机很轻松地重复执行相同的操作。循环是汇编之上的编程中最重要的概念之一。Matlab的循环有两个语言构造,一个是for循环,另一个是while循环。在Matlab中,for循环是最常用的循环结构。

然而,for循环在Matlab中是非常慢的。这是因为Matlab是一种解释性语言,而不是一种编译性语言。比如,立志于在科学计算领域开辟一“新”道路的Julia语言,其宣传的最重要特点之一就是for循环的速度非常快。

作为经常需要处理大量数据的科学计算工具,Matlab的for循环速度慢是一个很大的问题。因此,我们应该尽量避免使用for循环。

这里有一个说法可以特别地在这里提到,就是所谓的过早优化。过早优化是一种不好的编程习惯。在程序设计的初期,我们应该首先考虑代码的可读性和可维护性。然后,我们应该考虑代码的性能。然而在Matlab中,避免使用for循环并不意味着我们应该在一开始就考虑性能问题。通过采用下面我们提到的方式,编写的Matlab代码的可读性和可维护性也是非常强的。

1. For循环的基本知识

1.1 语法

在Matlab中,for循环的构造如下:

for i = 1:n% do something
end

这个构造同样也可以写成单行的形式:

for i = 1:n, statements, end

如果我们要对一个向量v进行操作,可以使用如下的for循环:

for i = 1:length(v)% do something
end

上面,1:n1:length(v)是for循环的索引。实际上,在Matlab中,:操作符会直接产生一个数组(Matlab最基本的数据结构),并且这个操作符还能够设定步长。比如,1:2:10会产生一个从1到10的数组,步长为2。

for语言构造采用 = 来迭代一个数组。所以,前一个例子里面,可以直接写成:

for vi = v% do something
end

1.2 break和continue

在For循环中,要避免改变索引变量的值来试图改变循环过程,for循环会直接覆盖对这个值的改变。这一点跟很多别的程序设计语言并不相同。

for i = 1:10i = 5; % 这个操作是无效的disp(i);
end

要中途退出一个循环,可以使用关键词break。要跳过当前循环的剩余部分,可以使用关键词continue。

运行一下上面的代码,会打印10个5。

1.3 循环二维数组

另外,我们都知道Matalb的2维数组(矩阵)是列先的,所以在for循环中,对矩阵的操作,应该是按列进行的。比如:

A = [1 2 3; 4 5 6; 7 8 9];
for Ai = Adisp(Ai);
end

上面的代码会打印出矩阵A的每一列。

在这里插入图片描述

Q: 如果上面的A是一个列向量,会怎么样?循环几次?

A: 循环一次。因为列向量是一个列,所以循环一次。

Q: 如果A是一个行向量呢?

A: 行向量有多少个元素,就循环多少次。

这里还挺有意思的,Matlab的for循环是按列进行的,所以对于一个矩阵,我们可以使用for循环来对每一列进行操作。这是一个很有用的特性。而一维数组,列向量和行向量,是一个维数为1的矩阵,所以对于这些数据,for循环的行为是具有一致性的。

2. 避免使用for循环

2.1 算法矩阵化

Matlab是一种矩阵化的语言。这意味着,Matlab的很多操作都是对整个矩阵进行的。与C、Java等语言不同,Matlab的很多操作都是对整个矩阵进行的,Matlab提供了大量针对矩阵进行计算的操作符和操作函数。这些操作符和函数都是高度优化过的,所以在Matlab中,我们应该尽量使用矩阵化的操作。这一点应该是Matlab的使用者在设计算法之初就应该考虑的。

把算法的步骤推导为矩阵的形式,可以为Matlab来实现提供很大的帮助。比如,我们要计算一个向量的平方和,可以使用for循环:

v = [1 2 3 4 5];
sumv = 0;
for vi = vsumv = sumv + vi^2;
end

这里需要注意,如果v是列向量,那么只会循环一次,得到奇怪的结果。(可以自己试试,并思考为什么)

我们可以使用矩阵化的操作:

v = [1 2 3 4 5];
sumv = v * v';

这里用到了行矩阵x列矩阵的乘法。

A m × n × B n × p = C m × p A 1 × n × B n × 1 = c \begin{split} & A_{m \times n} \times B_{n \times p} = C_{m \times p} \\ & A_{1 \times n} \times B_{n \times 1} = c \end{split} Am×n×Bn×p=Cm×pA1×n×Bn×1=c

2.2 计算向量化

Matlab还提供了另外一个非常强大的工具,那就是向量化。向量化就是把操作符应用到一个矩阵的每一个元素上。比如,计算向量平方和,可以用向量化的方法写成:

v = [1 2 3 4 5];
sumv = sum(v.^2);

这里用到一个向量化操作的函数sum和一个向量化算符.^。前者是对矩阵的每一个元素求和,后者是对矩阵的每一个元素进行平方。

Matlab提供了大量的向量化操作符和函数,这些函数和操作符都是高度优化过的

3. 数组的逻辑索引与arrayfun函数

Matlab还提供了一些数组的逻辑操作,这些操作也是高度优化过的。比如,我们要找出一个向量中所有大于5的元素:

v = [1 2 3 4 5 6 7 8 9];
vgt5 = v(v > 5);

这里用到了一个逻辑操作符>,这个操作符会产生一个逻辑数组,然后我们可以用这个逻辑数组来索引原数组。

通过逻辑数组索引,还能够实现很多高级的操作。比如,我们要把一个向量中所有的偶数都变成0:

v = [1 2 3 4 5 6 7 8 9];
v(v % 2 == 0) = 0;

这里用到了一个逻辑操作符==,这个操作符会产生一个逻辑数组,然后我们可以用这个逻辑数组来索引原数组。

可以通过比较复杂的函数来产生逻辑数组,然后用逻辑数组来索引原数组。这样,我们就可以把很多循环操作转化为矩阵化的操作。特别是,用逻辑判断的函数加上arrayfun函数,可以实现很多循环操作。

v = [1 2 3 4 5 6 7 8 9];
predicate = @(x) x > 5 && x < 8;
index = arrayfun(predicate, v);
v(index) = 0;

通过这样的办法,可以写出非常易读的代码,并且也可能会更快。

4. 总结

在Matlab中,for循环是非常慢的。我们应该尽量避免使用for循环。Matlab提供了大量的矩阵化操作符和函数,向量化操作符和函数,逻辑操作符和函数,这些操作符和函数都是高度优化过的。我们应该尽量使用这些操作符和函数,来代替for循环。

  1. 使用矩阵化的操作,从设计和推导算法的时候就采用矩阵来表示;
  2. 使用向量化的操作,对矩阵的每一个元素进行操作;
  3. 使用逻辑索引操作矩阵的部分元素;
  4. 可以考虑使用arrayfun函数来代替for循环。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/282273.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JetBrains全家桶激活,分享 GoLand 2024 激活的方案

大家好&#xff0c;欢迎来到金榜探云手&#xff01; GoLand 公司简介 JetBrains 是一家专注于开发工具的软件公司&#xff0c;总部位于捷克。他们以提供强大的集成开发环境&#xff08;IDE&#xff09;而闻名&#xff0c;如 IntelliJ IDEA、PyCharm、和 GoLand等。这些工具被…

总结mac下解决matplotlib中文显示问题的几种方法

一、前言&#xff1a; 使⽤matplotlib画图时&#xff0c;由于matplotlib默认没有中⽂&#xff0c;显⽰中文时会出现空⽩⼩⽅块。 二、方法&#xff1a; 2.1 matplotlib中使用SimHei字体 1&#xff09;进入终端后查看matplotlib的字体路径&#xff1a; $ python >>&g…

Linux学习-进程

目录 进程基本概念 进程相关命令 进程的创建 进程的调度 进程相关函数接口 进程的消亡 实例&#xff1a;创建九个子进程 目录 进程基本概念 进程相关命令 进程的创建 进程的调度 进程相关函数接口 进程的消亡 实例&#xff1a;创建九个子进程 exec函数…

Python脚本:用py处理PDF的五大功能

一、代码 【第三方库】3个 【Py版本】3.9 【使用前提】关闭所有的word文档 import os from datetime import datetime from docx2pdf import convert from pdf2docx import parse from PyPDF2 import PdfMerger from PyPDF2 import PdfReader,PdfWriter#将文件夹中的所有Wo…

C++语言学习(二)—— C++语言的基本知识

目录 一、面向对象的三个核心概念 二、C语言中的I/O口 三、C语言中的数据类型​​​​​​​ 3.1 逻辑类型 3.2 引用类型 3.2.1 引用作为函数参数 3.2.2 引用作为函数返回值 3.2.3 引用作为类成员 3.3 类类型 四、 C语言中的内联函数 五、 函数重载 六、 带默认形参…

四、分布式锁之自定义分布式锁

1、基本原理和实现方式对比 分布式锁&#xff1a;满足分布式系统或集群模式下多个进程可见并且互斥的锁。分布式锁的核心思想就是多线程都使用同一把锁&#xff0c;实现程序串行执行。 分布式锁需要具备的条件&#xff1a; 特性含义可见性多个线程都能感知到变化互斥性分布…

数据库系统概论-第16章 数据仓库与联机分析处理技术

概念性的介绍&#xff0c;一略而过&#xff0c;不重要。 16.1 数据仓库技术 16.2 联机分析处理技术 16.3 数据挖掘技术 16.4 大数据时代的新型数据仓库 16.5 小结

G - Find a way

题目分析 1.双重bfs,遍历两个起点求最短路再计算总和即可 2.唯一的坑点在于对于一个KFC&#xff0c;两人中可能有一个到不了&#xff0c;所以还要对到不了的点距离做处理 #include <bits/stdc.h> using namespace std; using ll long long; const int N 220;struct pos…

Linux/Ubuntu/Debian从控制台启动程序隐藏终端窗口

如果你想从终端运行应用程序但隐藏终端窗口. 你可以这样做&#xff1a; 在后台运行&#xff1a; 你只需在命令末尾添加一个与号 (&) 即可在后台运行它。 例如&#xff1a; your_command &将 your_command 替换为你要运行的命令。 这将在后台启动该命令&#xff0c…

Three.js基础入门介绍——【毕业季】Three.js动态相册

前言 岁月匆匆&#xff0c;又是一年毕业季&#xff0c;这次做个动态相册展示图片&#xff0c;放些有意思的内容&#xff0c;一起回忆下校园生活吧。 预期效果 相册展示和点选切换&#xff0c;利用相机旋转和移动来实现一个点击切图平滑过渡的效果。 实现流程 基本流程 1、搭…

【python】python汽车效能数据集—回归建模(源码+数据集)【独一无二】

&#x1f449;博__主&#x1f448;&#xff1a;米码收割机 &#x1f449;技__能&#x1f448;&#xff1a;C/Python语言 &#x1f449;公众号&#x1f448;&#xff1a;测试开发自动化【获取源码商业合作】 &#x1f449;荣__誉&#x1f448;&#xff1a;阿里云博客专家博主、5…

【学习】软件测试行业未来的发展趋势预测

近年来&#xff0c;随着中国数字经济的蓬勃发展&#xff0c;软件测试行业也迎来了新的春天。从早期的手工测试到自动化测试&#xff0c;再到持续集成和持续交付&#xff0c;中国的软件测试行业经历了快速的发展和变革。各行各业均对软件测试提出了更高的要求&#xff0c;尤其在…

将数据转换成xml格式的文档并下载

现在有一个实体类对象的集合&#xff0c;需要将它们转换为xml文档&#xff0c;xml文档就是标签集合的嵌套&#xff0c;例如一个学生类&#xff0c;有姓名、年龄等&#xff0c;需要转换成一下效果&#xff1a; <student><age>14</age><name>张三</na…

【Java】Oracle发布Java22最新版本

甲骨文&#xff08;ORACLE&#xff09;已经于2023年3月19日正式发布了最新版本的JDK&#xff0c;版本号&#xff1a;22 根据官方声明&#xff0c;Java 22 (Oracle JDK 22) 在性能、稳定性和安全性方面进行了数千种改进&#xff0c;包括对Java 语言、其API 和性能&#xff0c;以…

docker 哲学 - 网络桥接器、容器网络接口 、容器间的通信方式

1、解释 docker0 veth eth 2、vethXX 和 ethXX 是肯定一一对应吗 比如 eth1 对应 veth1 3、如果 A容器使用 默认创建方式 。定义他内部网络为 eth0&#xff0c;容器B使用 --network 连上 已创建的网络 172.89.2.1 。此时假设 B的 ip是 172.89.2.2 &#xff0c;容器网络接口是 e…

Godot 学习笔记(4):一切以场景为中心

文章目录 前言场景搭建新建子场景最简单的按钮事件 手动控制场景手动加载场景添加多个场景对象更快速的获取脚本对象 删除多个场景对象脚本命名的问题 总结 前言 Godot的场景是C#与Godot最后的中间连接。我们解决了场景的加载&#xff0c;我们基本可以保证C#和godot之间的彻底…

C++初阶:vector相关练习

目录 1. 只出现一次的数2. 杨辉三角3. 删除有序数组中的重复项4. 只出现一次的数II5. 只出现一次的数III6. 数组中出现次数超过一半的数7. 电话号码的字母组合&#xff08;多叉树遍历&#xff09; 1. 只出现一次的数 题目信息&#xff1a; 题目链接&#xff1a; 只出现一次的数…

工程信号的去噪和(分类、回归和时序)预测

&#x1f680;【信号去噪及预测论文代码指导】&#x1f680; 还为小论文没有思路烦恼么&#xff1f;本人专注于最前沿的信号处理与预测技术——基于信号模态分解的去噪算法和深度学习的信号&#xff08;回归、时序和分类&#xff09;预测算法&#xff0c;致力于为您提供最精确、…

ruoyi-nbcio-plus基于vue3的flowable增加开始节点的表单绑定修改

更多ruoyi-nbcio功能请看演示系统 gitee源代码地址 前后端代码&#xff1a; https://gitee.com/nbacheng/ruoyi-nbcio 演示地址&#xff1a;RuoYi-Nbcio后台管理系统 http://122.227.135.243:9666/ 更多nbcio-boot功能请看演示系统 gitee源代码地址 后端代码&#xff1a…

java.lang.String final

关于String不可变的问题&#xff1a;从毕业面试到现在&#xff0c;一个群里讨论的东西&#xff0c;反正码农面试啥都有&#xff0c;这也是我不咋喜欢面试代码&#xff0c;因为对于我而言&#xff0c;我并不喜欢这些面试。知道或不知道基本没啥含氧量&#xff0c;就是看看源代码…