当前位置:首页 > 文章列表 > Golang > Go教程 > 支持 Pascal 函数

支持 Pascal 函数

来源:dev.to 2024-07-09 19:45:54 0浏览 收藏

Golang小白一枚,正在不断学习积累知识,现将学习到的知识记录一下,也是将我的所得分享给大家!而今天这篇文章《支持 Pascal 函数》带大家来了解一下##content_title##,希望对大家的知识积累有所帮助,从而弥补自己的不足,助力实战开发!


支持 Pascal 函数

对于那些不遵循 POJ(JVM 上的 Pascal)的人来说,它是一个编译器,将子集从 Pascal 转换为 JASM(Java 程序集),以便我们可以使用 JVM 作为执行环境。

在上一篇文章中,我们在错误捕获、对 string 类型的关系运算符的支持以及定义(和使用)Pascal 过程.

的可能性方面进行了一些改进

在本出版物中,我们将介绍对 Pascal 函数(functions)的支持。不久之后我们就可以完成该项目的最后一个目标:从标准输入中读取一个数字并计算其阶乘。

当我们为 JVM 进行编译时,有必要详细说明这个令人难以置信的虚拟机的各个点的功能。因此,我多次详细介绍 JVM 的内部工作原理以及它的一些指令(操作码)。

支持 Pascal 函数(函数

到目前为止,我们有一种方法来定义和调用 Pascal 的过程。从此 PR 中还可以定义和调用 Pascal 的 函数.

在此提交中,实现了一个 Java 程序来了解 JVM 如何处理定义和调用函数。来自下面的Java程序:

公共类FunctionCall {
    公共静态无效主(字符串[] args){
        System.out.println("来自main的你好!");
        System.out.println(myMethod());
    }

    静态字符串 myMethod() {
        返回“来自 myMethod 的你好!”;
    }
}

当我们反汇编时,我们得到以下程序集

1: 公共类 FunctionCall {
2: 公共静态主([java/lang/String)V {
3:getstatic java/lang/System.out java/io/PrintStream
4:ldc“主站您好!”
5:调用虚拟java/io/PrintStream.println(java/lang/String)V
6:
7: getstatic java/lang/System.out java/io/PrintStream
8:invokestatic FunctionCall.myMethod()java/lang/String
9:调用虚拟java/io/PrintStream.println(java/lang/String)V
10:
11:返回
12:}
13:
14: 静态 myMethod()java/lang/String {
15:ldc“我的方法你好!”
16:
17:返回
18:}
19:}

通过这个例子,可以确定:

  • 为了调用方法,JVM 使用指令“invokestatic FunctionCall.myMethod()java/lang/String”(第 8 行),其中:
    • invokestatic 是接收要调用的方法的完整签名作为参数的指令;
    • FunctionCall 是类的名称;
    • myMethod()java/lang/String 是方法的完整签名及其参数(在本例中为无)和返回类型(在本例中为 java/lang/String);
  • 指令areturn(第17行)终止函数并将返回字符串留在堆栈上。

也就是说,来自下面的 Pascal 程序:

程序function_call_wo_params;

函数 myfunction :字符串;
开始
    myfunction := '来自 myfunction 的问候!';
结尾;

开始
    writeln('来自主程序的你好!');
    writeln(myfunction());
结尾。

POJ 已调整生成以下 JASM:

// POJ 0.1 生成的代码
公共类 function_call_wo_params {
    ;;函数 myfunction :字符串;
    静态 myfunction()java/lang/String {
        ldc“我的函数你好!”
        储存 100 ;;第 100 位保存函数的返回值
        加载 100 ;;堆栈函数返回值
        回报 ;;留下“来自我的功能的你好!”在堆栈中
    }

    ;;主要程序(主要)
    公共静态主([java / lang / String)V {
        ;; writeln('来自主程序的你好!');
        getstatic java/lang/System.out java/io/PrintStream
        ldc“主站您好!”
        invokevirtual java/io/PrintStream.print(java/lang/String)V
        getstatic java/lang/System.out java/io/PrintStream
        调用虚拟 java/io/PrintStream.println()V

        ;; writeln(myfunction());
        getstatic java/lang/System.out java/io/PrintStream
        invokestatic function_call_wo_params.myfunction()java/lang/String 
        invokevirtual java/io/PrintStream.print(java/lang/String)V
        getstatic java/lang/System.out java/io/PrintStream
        调用虚拟 java/io/PrintStream.println()V

        返回
    }
}

最细心的人一定注意到了上面的“astore 100”并想:

  • 为什么要将函数返回值存储在局部变量中?这是因为在 Pascal 中,函数的返回值在函数执行期间可以设置 N 次,但我们在 JVM 中只能堆栈一次结果;
  • 为什么排在第100位?函数或过程的局部变量从位置 0 开始,因此任意选择位置 100 来存储返回值;
  • 但是是否可以进行优化,以便在本例中仅生成指令ldc“Hello from myfunction!”,然后生成指令areturn?是的,是的,但是 POJ 没有实现市场编译器中存在的优化阶段,这可能在未来实现。

此提交实现了对符号表中的“function”类型和解析器的支持。

在上面的示例中,函数没有参数。在此提交中,实现了带有参数的函数的预期结果。通过下面的 Pascal 程序:

程序 function_call_with_two_params;

函数 addvalues(value1, value2: 整数) : 整数;
开始
    添加值:=值1+值2;
结尾;

开始
    writeln('2+4=', addvalues(2, 4));
结尾。

POJ 正确生成了以下 JASM:

// POJ 0.1 生成的代码
公共类 function_call_with_two_params {
    ;;函数 addvalues(value1, value2: 整数) : 整数;
    静态加值(I,I)I {
        ;;添加值:=值1+值2;
        加载0
        加载1
        我加 
        这是 100
        加载100

        返回 
    }

    ;;主要程序
    公共静态主([java / lang / String)V {
        ;; writeln('2+4=', ...);
        getstatic java/lang/System.out java/io/PrintStream
        LDC“2+4=”
        invokevirtual java/io/PrintStream.print(java/lang/String)V
        getstatic java/lang/System.out java/io/PrintStream

        ;;这里的代码调用 addvalues(2, 4)
        西普什 2
        西普什 4
        invokestatic function_call_with_two_params.addvalues(I, I)I 

        ;;这里的代码调用 writeln 并返回 addvalues
        调用虚拟 java/io/PrintStream.print(I)V
        getstatic java/lang/System.out java/io/PrintStream
        调用虚拟 java/io/PrintStream.println()V

        返回
    }
}

下一步

在下一篇文章中,我们将讨论上下文、发现的错误、嵌套句子、数据输入,并总结该项目的最后一个目标:递归计算阶乘。

完整的项目代码

包含项目完整代码和文档的存储库在这里。

理论要掌握,实操不能落!以上关于《支持 Pascal 函数》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

版本声明
本文转载于:dev.to 如有侵犯,请联系study_golang@163.com删除
案例(一)-KisFlow-Golang流实时计算-快速入门指南案例(一)-KisFlow-Golang流实时计算-快速入门指南
上一篇
案例(一)-KisFlow-Golang流实时计算-快速入门指南
在 React 中构建可重用的列表组件
下一篇
在 React 中构建可重用的列表组件
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    508次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    497次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 茅茅虫AIGC检测:精准识别AI生成内容,保障学术诚信
    茅茅虫AIGC检测
    茅茅虫AIGC检测,湖南茅茅虫科技有限公司倾力打造,运用NLP技术精准识别AI生成文本,提供论文、专著等学术文本的AIGC检测服务。支持多种格式,生成可视化报告,保障您的学术诚信和内容质量。
    103次使用
  • 赛林匹克平台:科技赛事聚合,赋能AI、算力、量子计算创新
    赛林匹克平台(Challympics)
    探索赛林匹克平台Challympics,一个聚焦人工智能、算力算法、量子计算等前沿技术的赛事聚合平台。连接产学研用,助力科技创新与产业升级。
    112次使用
  • SEO  笔格AIPPT:AI智能PPT制作,免费生成,高效演示
    笔格AIPPT
    SEO 笔格AIPPT是135编辑器推出的AI智能PPT制作平台,依托DeepSeek大模型,实现智能大纲生成、一键PPT生成、AI文字优化、图像生成等功能。免费试用,提升PPT制作效率,适用于商务演示、教育培训等多种场景。
    120次使用
  • 稿定PPT:在线AI演示设计,高效PPT制作工具
    稿定PPT
    告别PPT制作难题!稿定PPT提供海量模板、AI智能生成、在线协作,助您轻松制作专业演示文稿。职场办公、教育学习、企业服务全覆盖,降本增效,释放创意!
    108次使用
  • Suno苏诺中文版:AI音乐创作平台,人人都是音乐家
    Suno苏诺中文版
    探索Suno苏诺中文版,一款颠覆传统音乐创作的AI平台。无需专业技能,轻松创作个性化音乐。智能词曲生成、风格迁移、海量音效,释放您的音乐灵感!
    109次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码