matlab递归函数怎么写 编写一个递归函数

matlab递归函数怎么写 编写一个递归函数

前言

本篇文章主要是记录一下在 GScript 中实现递归调用时所遇到的坑,类似的问题在中文互联网上我几乎没有找到相关的内容,所以还是很有必要记录一下。

在开始之前还是简单介绍下本次更新的 GScript v0.0.9 所包含的内容:

  • 支持可变参数
  • 优化 append 函数语义
  • 优化编译错误信息
  • 最后一个就是支持递归调用

先看第一个可变参数:

//formats according to a format specifier and writes to standard output. printf(string format, any ...a){} //formats according to a format specifier and returns the resulting string. string sprintf(string format, any ...a){}

以上是随着本次更新新增的两个标准函数,均支持可变参数,其中使用 … 表示可变参数,调用时如下:

printf("hello %s ","123"); printf("hello-%s-%s ","123","abc"); printf("hello-%s-%d ","123",123); string format = "this is %s "; printf(format, "gscript"); string s = sprintf("nice to meet %s", "you"); assertEqual(s,"nice to meet you");

与大部分语言类似,可变参数本质上就是一个数组,所以可以拿来循环遍历:

int add(string s, int ...num){ println(s); int sum = 0; for(int i=0;i<len(num);i++){ int v = num; sum = sum+v; } return sum; } int x = add("abc", 1,2,3,4); println(x); assertEqual(x, 10);


// appends "v" to the end of a array "a" append(any a, any v){}

之后是优化了内置函数 append() 的语义,本次优化来自于 issue12 的建议: https://github.com/crossoverJie/gscript/issues/12

// Before int a={1,2,3}; println(a); println(); a = append(a,4); println(a); // Output: [1 2 3 4] // Now int a={1,2,3}; println(a); println(); append(a,4); int b = a; assertEqual(4, b); println(a); // Output: [1 2 3 4]

现在 append 之后不需要再重新赋值,也会追加数据,优化后这里看起来是一个值/引用传递的问题,但其实底层也是值传递,只是在语法上增加了这样的语法糖,帮使用者重新做了一次赋值。


之后是新增了编译错误信息提示,比如下面这段代码:

a+2; b+c;

使用没有声明的变量,现在会直接编译失败:

1:0: undefined: a 2:0: undefined: b 2:2: undefined: c

class T{} class T{} // output: 2:0: class T redeclared in this block

重复声明之类的语法错误也有相关提示。


最后一个才是本次讨论的重点,也就是递归函数的支持。

int num(int x,int y){ if (y==1 || y ==x) { return 1; } int v1 = num(x - 1, y - 1); return c; }

再上一个版本中 int v1 = num(x – 1, y – 1); 这行代码是不会执行的,具体原因后文会分析。

现在利用递归便可以实现类似于打印杨辉三角之类的程序了:

int num(int x,int y){ if (y==1 || y ==x) { return 1; } int v1 = num(x - 1, y - 1); int v2 = num(x - 1, y); int c = v1+v2; // int c = num(x - 1, y - 1)+num(x - 1, y); return c; } printTriangle(int row){ for (int i = 1; i <= row; i++) { for (int j = 1; j <= row - i; j++) { print(" "); } for (int j = 1; j <= i; j++) { print(num(i, j) + " "); } println(""); } } printTriangle(7); // output: 1 1 1 1 2 1 1 3 3 1 1 4 6 4 1 1 5 10 10 5 1 1 6 15 20 15 6 1

函数中的 return

int num(int x,int y){ if (y==1 || y ==x) { return 1; } int v1 = num(x - 1, y - 1); return c; }

现在我们来看看这样的代码为什么执行完 return 1 之后就不会执行后边的语句了。

其实在此之前我首先解决的时候函数 return 后不能执行后续 statement 的需求,其实正好就是上文提到的逻辑,只是这里是递归而已。

先把代码简化一下方便分析:

int f1(int a){ if (a==10){ return 10; } println("abc"); }

当参数 a 等于 10 的时候确实不能执行后续的打印语句了,那么如何实现该需求呢?

以正常人类的思考方式:当我们执行完 return 语句的时候,就应该标记该语句所属的函数直接返回,不能在执行后续的 statement。

可是这应该如何实操呢?

其实看看 AST 就能明白了:

matlab递归函数怎么写 编写一个递归函数

当碰到 return 语句的时,会递归向上遍历语法树,标记上所有 block 节点表明这个 block 后续的语句不再执行了,同时还得把返回值记录下来。

这样当执行到下一个 statement 时,也就是 println("abc"); 则会判断他所属的 block 是否有被标记,如果有则直接返回,这样便实现了 return 语句不执行后续代码。

部分实现代码如下:

// 在 return 的时候递归向上扫描所有的 Block,并打上标记,用于后面执行 return 的时候直接返回。 func (v *Visitor) scanBlockStatementCtx(tree antlr.ParseTree, value interface{}) { context, ok := tree.(*parser.BlockContext) if ok { if v.blockCtx2Mark == nil { v.blockCtx2Mark = make(map[*parser.BlockContext]interface{}) } v.blockCtx2Mark = value } if tree.GetParent() != nil { v.scanBlockStatementCtx(tree.GetParent().(antlr.ParseTree), value) } }

matlab递归函数怎么写 编写一个递归函数

源码地址: https://github.com/crossoverJie/gscript/blob/793d196244416574bd6be641534742e57c54db7a/visitor.go#L182

递归的问题

但同时问题也来了,就是递归的时候也不会执行后续的递归代码了。

其实解决问题的方法也很简单,就是在判断是否需要直接返回那里新增一个条件,这个 block 中不存在递归调用。

所以我们就得先知道这个 block 中是否存在递归调用。

整个过程有以下几步:

  • 编译期:在函数声明处记录下函数与当前 context 的映射关系。
  • 编译期:扫描 statement 时,取出该 statement 的 context 所对应的函数。
  • 编译期:扫描到的 statement 如果是一个函数调用,则判断该函数是否为该 block 中的函数,也就是第二步取出的函数。
  • 编译期:如果两个函数相等,则将当前 block 标记为递归调用。
  • 运行期:在刚才判断 return 语句处,额外多出判断当前 block 是否为递归调用,如果是则不能返回。

部分代码如下:

matlab递归函数怎么写 编写一个递归函数

matlab递归函数怎么写 编写一个递归函数

https://github.com/crossoverJie/gscript/blob/3e179f27cb30ca5c3af57b3fbf2e46075baa266b/resolver/ref_resolver.go#L70

总结

这里的递归调用其实卡了我挺长时间的,思路是有的,但是写出来的代码总是和预期不符,当天晚上坐在电脑面前到凌晨两三点,百思不得其解。

最后受不了上床休息的时候,突然一个灵光乍现让我想到了解决方案,于是第二天起了个早床赶忙实践,还真给解决了。

所以有些时候碰到棘手问题时给自己放松一下,往往会有出其不意的效果。

最后是目前的递归在某些情况下性能还有些问题,后续会尽量将这些标记过程都放在编译期,编译慢点没事,但运行时慢那就有问题了。

之后还会继续优化运行时的异常,目前是直接 panic,堆栈也没有,体感非常不好;欢迎感兴趣的朋友试用反馈bug。

源码地址:

https://github.com/crossoverJie/gscript

本文【matlab递归函数怎么写 编写一个递归函数】由作者: 主键 提供,本站不拥有所有权,只提供储存服务,如有侵权,联系删除!
本文链接:https://www.cuoshuo.com/blog/4468.html

(0)
上一篇 2023-03-14 08:04:14
下一篇 2023-03-14 08:14:28

相关推荐

  • 二进制文件损坏怎么修复

    如果您不想将工件上传到公共存储库,或者如果您在未连接到公共存储库的私有网络上工作,则需要能够将依赖项离线存储在私有存储库中并在本地推送或拉取它们。 根据维基百科,二进制存储库管理器 (BRM)是“一种软件工具,旨在优化软件开发中使用和生成的二进制文件的下载和存储”,例如 .jar、.tar 或 .zip 档案。作为大多数DevOps 工具链的关键组件,BRM…

    2023-03-11
    300
  • 电脑显示器花屏自修方法(电脑显示器花屏横条纹闪)

    显示器花屏是一个让人非常头疼的问题,久而久之可能会影响到电脑的正常使用。花屏可以简单理解为显示器内出现了一些杂乱无章的彩色点和条纹,影响着显示器的视觉效果。那么显示器花屏能修吗? 1、原因分析:显示器花屏的原因有很多,它们可能是由于显示器的芯片出现了错误或者其他故障所导致的,也可能是由于显卡和显示器之间的适配问题所导致的,甚至可能是显示器外壳内部聚集了大量灰…

    2023-03-13
    500
  • 制作html简易个人主页

    HelloWebUI 分享多主题、多语言、响应式web网页模板 移动端语言切换展示 IPad/PC 多主题展示 源码 <!DOCTYPE html> <html> <head> <meta charset="UTF-8" /> <meta name="viewport&#03…

    2023-03-12
    300
  • 关系型数据库的基本原理(数据库系统的组成和特点)

    我们所熟知的数据库一般都是关系型数据库,比如Oracle、Sql Server、DB2和Mysql等。Oracle一般用在电信公司,Sql Server可以在中小型企业或零售公司寻觅其踪影,DB2一般配合IBM的大型机(如OS390),只有高大上的银行能够用得起来,而Mysql因其开源和支持高可用集群,经常在淘宝等网站亮其身份。 谈到数据库,最基本的莫过于对…

    2023-03-15
    100
  • spring框架是前端还是后端_web前端三大主流框架

    Web开发人员应该在实践中快速、灵活、可靠地完成任务。由于最近引入的框架或开发工具通常会产生更好的结果,开发人员必须不断提升他们的技能和新技术知识。但是这些web开发框架是什么呢? 为了方便起见,我们将Web开发框架列表分为两部分:前端框架、后端框架。 前端Web开发框架 1. Angular Angular是一个开源的javascript框架,旨在创建单页…

    2023-03-10
    800
  • matlab中取整数和小数部分,c语言整型变量四舍五入吗

    首先说明一点,MATLAB软件是基于C语言编写开发的。因为C语言的稳定性是其他计算机编程语言所无法比拟的,所以大部分工程软件的开发都采用C语言。在C语言中,没有提供丰富的函数库;而在MATLAB语言中,已经为用户编写好大量常用函数,诸如三角函数、反三角函数、取余、取整、绝对值、开方、求和及求角等常用函数。此外,MATLAB语言继承并发展了C语言中的运算符,除…

  • sort排序函数

    功能介绍: sort命令 是在Linux里非常有用,它将文件进行排序,并将排序结果标准输出。sort命令既可以从特定的文件,也可以从stdin中获取输入。 语法格式:sort 常用参数: -b忽略每行前面开始出的空格字符。 -c检查文件是否已经按照顺序排序。 -d排序时,处理英文字母、数字及空格字符外,忽略其他的字符。 -f排序时,将小写字母视为大写字母。 …

    2023-03-08
    1200
  • 要取消已设置的超级链接 怎么快速取消超链接

    Excel是我们经常用的办公软件,我们在Excel表格中输入网址或邮箱地址时,Excel会将它们自动转化为超链接。如果我们不需要设置超链接,那我们该怎么办呢?今天,小编就教各位Excel中批量取消超链接的小技巧,大家一起来学习吧!首先,打开我们的Excel表格,在上面输入一些网址和邮箱地址,就会发现它们已经被自动转为了超链接; 然后,我们批量选中需要取消超链…

    2023-03-12
    300
  • java数据库课程设计报告

    2,数据库设计 2.1 数据库设计简介 软件的研发步骤 数据库设计概念 数据库设计就是根据业务系统的具体需求,结合我们所选用的DBMS,为这个业务系统构造出最优的数据存储模型。 建立数据库中的表结构以及表与表之间的关联关系的过程。 有哪些表?表里有哪些字段?表和表之间有什么关系? 数据库设计的步骤 需求分析(数据是什么? 数据具有哪些属性? 数据与属性的特点…

    2023-03-14
    100
  • informix replace函数

    【功能】 REPLACE函数使用其他文本字符串并根据所指定的字符数替换某文本字符串中的部分文本。无论默认语言设置如何,始终将每个字符(不管是单字节还是双字节)按1计数。 【语法】 REPLACE(old_text,start_num,num_chars,new_text) 【参数】 old_text:表示要替换其部分字符的文本。 start_num:指定替换…

    2023-03-22
    000

发表回复

登录后才能评论
返回顶部
错说博客上线啦!