编译原理Lab. 1 初代编译器实验说明和要求
目录
- Lab. 1 初代编译器实验说明和要求
- 一、初代编译器功能描述
- 二、初代编译器文法要求
- 三、初代编译器测试样例
- 四、初代编译器提交要求
- 五、初代编译器实验测试框架说明
- 代码与思路
Lab. 1 初代编译器实验说明和要求
一、初代编译器功能描述
初代编译器将 C 语言顺序语句序列翻译为等价的汇编程序,所输出的汇编程序符合 x86 汇编语言格式要求,能够被后续的汇编器翻译为可执行程序运行。
二、初代编译器文法要求
初代编译器能够处理的文法如下所示:
关键字:int、 return
标识符:单个英文字母
常量:十进制整型,如 1、223、10 等
操作符:=、+、-、*、/、(、)
分隔符:;
语句:表达式语句、赋值语句,其中表达式语句包含括号及括号嵌套;
三、初代编译器测试样例
测试用例难度分为两个等级:其中第一个等级每个表达式中操作符优先级相同,且无括号;第二个等级同一个表达式中会有不同优先级,且包含有嵌套的括号。测试用例中,第一个等级测试用例占比 90%,第二个等级的测试用例占比 10%。
所有输入测试样例文件中单词之间均由空格或者回车分隔,输入文件中可能存在多个连续的空格或者回车。
评分依据 return 的值是否符合预期。
等级一输入样例:
int a ;
int b ;
int d ;
a = 1 ;
b = 2 ;
d = a + b ;
return d ;
(预期返回值为 3)
等级一输出样例 x86:
mov DWORD PTR [ebp-4], 0 # int a
mov DWORD PTR [ebp-8], 0 # int b
mov DWORD PTR [ebp-12], 0 # int d
mov DWORD PTR [ebp-4], 1 # a = 1
mov DWORD PTR [ebp-8], 2 # b = 2
mov eax, DWORD PTR [ebp-4] # d = a + b
push eax
mov eax, DWORD PTR [ebp-8]
push eax
pop ebx
pop eax
add eax, ebx
push eax
pop eax
mov DWORD PTR [ebp-12], eax
mov eax, DWORD PTR [ebp-12] # return d
等级二输入样例:
int a ;
int b ;
int c ;
int d ;
a = 1 ;
b = 2 ;
c = 3 ;
d = ( a + b * 2 ) / c – 3 ;
return d ;
(预期返回值为-2)
等级二输出样例 x86:
用例 2 的期待输出: (你的输出中不必输出#行的内容)
# int a ;
mov DWORD PTR [ebp-4], 0
# int b ;
mov DWORD PTR [ebp-8], 0
# int c ;
mov DWORD PTR [ebp-12], 0
# int d ;
mov DWORD PTR [ebp-16], 0
# a = 1 ;
mov DWORD PTR [ebp-4], 1
# b = 2 ;
mov DWORD PTR [ebp-8], 2
# c = 3 ;
mov DWORD PTR [ebp-12], 3
# d = ( a + b * 2 ) / c - 3
mov eax, DWORD PTR [ebp-4]
push eax
mov eax, DWORD PTR [ebp-8]
push eax
mov eax, 2
push eax
pop ebx
pop eax
imul eax, ebx
push eax
pop ebx
pop eax
add eax, ebx
push eax
mov eax, DWORD PTR [ebp-12]
push eax
pop ebx
pop eax
cdq
idiv ebx
push eax
mov eax, 3
push eax
pop ebx
pop eax
sub eax, ebx
push eax
pop eax
mov DWORD PTR [ebp-16], eax
# return d
mov eax, DWORD PTR [ebp-16]
四、初代编译器提交要求
实现语言:C++(语言标准 c++14)
编译环境:g++ -11
测试环境:gcc -11
提交内容:单个 cpp 源文件,文件名称为 compilerlab1.cpp
输入输出:实现的编译器有一个命令行参数,用于指明输入文件路径,编译器从该路径读取源码,并向 stdout 输出编译结果。
注:g++用于编译你提交的编译器实验源码,gcc 用于将你的编译器实验输出的 x86 汇编码编译成可执行文件,用于测试。
五、初代编译器实验测试框架说明
为了方便测试,实验提供了一个测试框架,用于测试你的编译器实验。
x86 汇编的测试框架:
.intel_syntax noprefix # 使用 Intel 语法
.global main # 声明 main 函数为全局符号,这使得链接器能够识别程序
的入口点。
.extern printf # 声明外部函数 printf,表示该函数在其他地方定义,通
常是 C 标准库中。
.data # 开始数据段,用于定义程序中的初始化数据。
format_str:.asciz "%d\n" # 定义一个用于 printf 的格式字符串,输出整数并换行。
.text # 开始代码段,包含程序的实际指令。
main:push ebp # 将基指针寄存器 ebp 的当前值压入堆栈,保存上一个函
数栈帧的基指针mov ebp, esp # 将栈指针 esp 的值复制到基指针 ebp ,设置新的栈帧
基指针sub esp, 0x100 # 从栈指针 esp 减去 256 字节,为局部变量分配栈空间
##################################################
##
## 你的编译器实验输出的 x86 汇编码将被插入到这里
##
################################################### 打印 d (当前 eax 的值)push eax # 将结果 (eax 的值) 作为 printf 的参数push offset format_str # 将格式字符串的地址作为 printf 的参数call printf # 调用 printf 函数add esp, 8 # 清理栈# 恢复 eax 的值并退出 mainpop eaxleaveret
你的编译器实验输出的 x86 汇编码将会被插入到上述框架中。
为了便于评测,本实验框架将会自动调用 C 库的 printf 函数,输出你的编译器实验输出的 x86 汇编码中的返回值(即 eax 的内容)。
代码与思路
考虑到代码会查重,以及隔壁3月16号大哥的惨案,我不会给出具体的代码。(至少要过半年我才会补上代码)
请同学们自己完成代码。
我会给出一些实验注意事项,该部分主要阐述本人在做该实验时踩过的坑,为大家提供一部分参考。
- 注意按照平台要求的命令行输入输出,否则提交无效
- 关于测试用例8:需要处理可能出现的回车,可能出现一条语句中间被回车截断,如
int a ;
a
=
1
; // 一条语句 a = 1 ; 被拆成了多行输入
return a
;
// 还可能会出现同一行多条语句,例如:
int b ; int c ; b = 1 ; c = 2 ;
请注意以上情况。
- 关于测试用例3和7:注意处理直接赋变量的值的语句,如
int a ;
int b ;
a = 1 ; // 对于这一句,直接把1存入a即可,即 mov DWORD PTR [ebp-4], 1
b = a ; // 对于这一句,容易写成把a的值直接mov到b中;然而这样做是错的,要先从a取出放到eax,再从eax取出放到b
return b ;
这里再贴两组用例供大家测试。
输入用例 3: (试试直接赋值,顺便试试大写字母)
int a ;
int A ;
int B ;
a = 1 ;
A = a ;
B = A / a ;
return B ;
用例 3 的期待输出: (你的输出不必像我这里特意搞空行)
mov DWORD PTR [ebp-4], 0mov DWORD PTR [ebp-8], 0mov DWORD PTR [ebp-12], 0mov DWORD PTR [ebp-4], 1mov eax, DWORD PTR [ebp-4]
mov DWORD PTR [ebp-8], eaxmov eax, DWORD PTR [ebp-8]
push eax
mov eax, DWORD PTR [ebp-4]
push eax
pop ebx
pop eax
cdq
idiv ebx
push eax
pop eax
mov DWORD PTR [ebp-12], eaxmov eax, DWORD PTR [ebp-12]
输入用例 4: (莫名其妙的换行)
int
p
; int q ; p =
1
;
q
= p
+
p ; return
q
;
用例 4 的期待输出: (你的输出不必像我这里特意搞空行)
mov DWORD PTR [ebp-4], 0mov DWORD PTR [ebp-8], 0mov DWORD PTR [ebp-4], 1mov eax, DWORD PTR [ebp-4]
push eax
mov eax, DWORD PTR [ebp-4]
push eax
pop ebx
pop eax
add eax, ebx
push eax
pop eax
mov DWORD PTR [ebp-8], eaxmov eax, DWORD PTR [ebp-8]
以上。
someday,我会把我的代码补充进来,顺便在git上再发一份
但不是现在:)
发布后的一小时后补充
我忘记写思路了2333
你可以直接 if :
- 如果遇到了
int,就输出mov DWORD PTR [ebp-%d], 0 - 如果遇到了
=,就说明是等式,可能是赋值,或可能是算式 - 如果遇到了
return,就输出mov eax, DWORD PTR [ebp-%d],并结束程序
对于遇到了 =:
- 右边只有一个数字
- 右边只有一个字母
- 如果不是前两者,说明是算式
如果是算式:
- 用我们以前学过的和栈有关的知识去做
- 下面是我的处理手段
- 定义两个栈:
op用来存储符号,v用来存储值 - 如果是数字
- 如果是字母
- 如果既不是数字也不是字母,那就说明是符号:
- 如果是
( - 如果是
) - 如果是
*或/ - 如果是
+或- - 当然,你也可以先转换成前缀表或者后缀表达式,再用相应的方法处理,自己去百度吧
- 如果是
- 定义两个栈:
你也可以使用正则表达式
如果这些方法都不合你的胃口,那你问问身边的人,方法真的很多。
我就点到为止吧,剩下的靠你们自己了(笔芯)。
祝大家早日AC
如果有什么思路上的问题或者编译上的问题总之跟这个实验有关的都可以问,我要是有空会回答的。
相关文章:
编译原理Lab. 1 初代编译器实验说明和要求
目录 Lab. 1 初代编译器实验说明和要求一、初代编译器功能描述二、初代编译器文法要求三、初代编译器测试样例四、初代编译器提交要求五、初代编译器实验测试框架说明 代码与思路 Lab. 1 初代编译器实验说明和要求 一、初代编译器功能描述 初代编译器将 C 语言顺序语句序列翻…...
python判断工作日,节假日
一、概述 需要判断一个日期是否为工作日,节假日。 找到一个现成的插件,蛮好用的。 插件介绍 https://pypi.org/project/chinesecalendar/ 判断某年某月某一天是不是工作日/节假日。 支持 2004年 至 2020年,包括 2020年 的春节延长。 兼容…...
练习4-权重衰减(李沐函数简要解析)
环境:练习1的环境 代码详解 0.导入库 import torch from torch import nn from d2l import torch as d2l1.初始化数据 这里初始化出train_iter test_iter 可以查一下之前的获取Fashion数据集后的数据格式与此对应 n_train, n_test, num_inputs, batch_size 20, 100, 200, …...
websocket 中 request-line 中的URI编码问题
首先,request-line组成如下: Request-Line Method SP Request-URI SP HTTP-Version CRLF 在 rfc6455 规范的 5.1.2 Request-URI 中,有这样的描述: The Request-URI is transmitted in the format specified in section 3.2.1. …...
为何ChatGPT日耗电超50万度?
看新闻说,ChatGPT每天的耗电量是50万度,国内每个家庭日均的耗电量不到10度,ChatGPT耗电相当于国内5万个家庭用量。 网上流传,英伟达创始人黄仁勋说:“AI的尽头是光伏和储能”,大佬的眼光就是毒辣ÿ…...
__init__.py 的作用
在 Python 中,包含一个名为 __ init __.py 的文件的目录被称为一个包(package)。 __ init __.py 文件的作用有以下几点: 指示包含该文件的目录是一个 Python 包:当 Python 导入一个包时,会查找该包所在目录…...
Redis到底是多线程还是单线程?
Redis6.0之前:是单线程模式。 Redis6.0之后:Redis的IO线程是多线程,worker线程是单线程。 Redis6.0之前:单线程 Redis6.0之后:Redis的IO线程是多线程,worker线程是单线程。...
JAVA 100道题(18)
18.实现一个除法运算的方法,能够处理被除数为零的情况,并抛出异常。 在Java中,你可以创建一个除法运算的方法,该方法接受两个整数作为参数,分别代表被除数和除数。如果被除数为零,你可以抛出一个自定义的异…...
【C++】每日一题 137 只出现一次的数字
给你一个整数数组 nums ,除某个元素仅出现 一次 外,其余每个元素都恰出现 三次 。请你找出并返回那个只出现了一次的元素。 你必须设计并实现线性时间复杂度的算法且使用常数级空间来解决此问题。 #include <vector>int singleNumber(std::vecto…...
RAG进阶笔记:RAG进阶
1 查询/索引部分 1.1 层次索引 创建两个索引——一个由摘要组成,另一个由文档块组成分两步进行搜索:首先通过摘要过滤出相关文档,接着只在这个相关群体内进行搜索 1.2 假设性问题 让LLM为每个块生成一个假设性问题,并将这些问…...
《论文阅读》带边界调整的联合约束学习用于情感原因对提取 ACL 2023
《论文阅读》带边界调整的联合约束学习用于情感原因对提取 前言简介Clause EncoderJoint Constrained LearningBoundary Adjusting损失函数前言 亲身阅读感受分享,细节画图解释,再也不用担心看不懂论文啦~ 无抄袭,无复制,纯手工敲击键盘~ 今天为大家带来的是《Joint Cons…...
【微服务】接口幂等性常用解决方案
一、前言 在微服务开发中,接口幂等性问题是一个常见却容易被忽视的问题,同时对于微服务架构设计来讲,好的幂等性设计方案可以让程序更好的应对一些高并发场景下的数据一致性问题。 二、幂等性介绍 2.1 什么是幂等性 通常我们说的幂等性&…...
RocketMQ学习笔记:零拷贝
这是本人学习的总结,主要学习资料如下 马士兵教育rocketMq官方文档 目录 1、零拷贝技术1.1、什么是零拷贝1.2、mmap()1.3、Java中的零拷贝 1、零拷贝技术 1.1、什么是零拷贝 使用传统的IO,从硬盘读取数据然后发送到网络需要经过四个步骤。 通过DMA复…...
3.26日总结
1.Fliptile Sample Input 4 4 1 0 0 1 0 1 1 0 0 1 1 0 1 0 0 1 Sample Output 0 0 0 0 1 0 0 1 1 0 0 1 0 0 0 0 题意:在题目输入的矩阵,在这个矩阵的基础上,通过最少基础反转,可以将矩阵元素全部变为0,如果不能达…...
应用日志集成到ElasticSearch
1、阿里云sls平台集成日志 阿里sls集成日志步骤 2、filebeat 收集到指定es 安装docker容器 Docker安装 拉取镜像: docker pull elastic/filebeat:7.5.1启动: docker run -d --namefilebeat elastic/filebeat:7.5.1拷贝容器中的数据文件到宿主机&a…...
MySQL多表联查函数
1 多表联查 1.1 表之间的关系 表和表的关系有: 一对一 老公 --> 老婆 , 人 ---> 身份证/户口本 一对多 皇帝 --> 妻妾 , 人 ---> 房/车 多对多 订单 --> 商品 1.2 合并结果集 合并结果集,是将多表查询的结果纵向合并 语法: select field1,field2 from t1 un…...
JAVAEE—实现多线程版本的定时器
文章目录 什么是定时器定时器的概念定时器的简单应用和介绍代码示例 定时器的代码解析定时器在执行任务的时候是创建了一个线程去执行吗?为什么叫做扫描线程呢?执行完任务之后代码就暂停了不自动结束吗? 手撕定时器demo相对时间与绝对时间Myt…...
KY228 找位置(用Java实现)
描述 对给定的一个字符串,找出有重复的字符,并给出其位置,如:abcaaAB12ab12 输出:a,1;a,4;a,5;a,10,b,2&…...
物联网边缘网关有哪些优势?-天拓四方
随着物联网技术的快速发展,越来越多的设备接入网络,数据交互日益频繁,对数据处理和传输的要求也越来越高。在这样的背景下,物联网边缘网关应运而生,以其低延迟、减少带宽消耗、提高数据质量和安全性等优势,…...
【C++】6-2 交换函数2 分数 10
6-2 交换函数2 分数 10 全屏浏览 切换布局 作者 刘利 单位 惠州学院 根据题目需求,编写一个交换函数Swap。 裁判测试程序样例: #include <iostream> using namespace std; class pen{private:string brand;string color;double price;publi…...
云计算——弹性云计算器(ECS)
弹性云服务器:ECS 概述 云计算重构了ICT系统,云计算平台厂商推出使得厂家能够主要关注应用管理而非平台管理的云平台,包含如下主要概念。 ECS(Elastic Cloud Server):即弹性云服务器,是云计算…...
Prompt Tuning、P-Tuning、Prefix Tuning的区别
一、Prompt Tuning、P-Tuning、Prefix Tuning的区别 1. Prompt Tuning(提示调优) 核心思想:固定预训练模型参数,仅学习额外的连续提示向量(通常是嵌入层的一部分)。实现方式:在输入文本前添加可训练的连续向量(软提示),模型只更新这些提示参数。优势:参数量少(仅提…...
Java 语言特性(面试系列1)
一、面向对象编程 1. 封装(Encapsulation) 定义:将数据(属性)和操作数据的方法绑定在一起,通过访问控制符(private、protected、public)隐藏内部实现细节。示例: public …...
云启出海,智联未来|阿里云网络「企业出海」系列客户沙龙上海站圆满落地
借阿里云中企出海大会的东风,以**「云启出海,智联未来|打造安全可靠的出海云网络引擎」为主题的阿里云企业出海客户沙龙云网络&安全专场于5.28日下午在上海顺利举办,现场吸引了来自携程、小红书、米哈游、哔哩哔哩、波克城市、…...
【大模型RAG】Docker 一键部署 Milvus 完整攻略
本文概要 Milvus 2.5 Stand-alone 版可通过 Docker 在几分钟内完成安装;只需暴露 19530(gRPC)与 9091(HTTP/WebUI)两个端口,即可让本地电脑通过 PyMilvus 或浏览器访问远程 Linux 服务器上的 Milvus。下面…...
React Native在HarmonyOS 5.0阅读类应用开发中的实践
一、技术选型背景 随着HarmonyOS 5.0对Web兼容层的增强,React Native作为跨平台框架可通过重新编译ArkTS组件实现85%以上的代码复用率。阅读类应用具有UI复杂度低、数据流清晰的特点。 二、核心实现方案 1. 环境配置 (1)使用React Native…...
渲染学进阶内容——模型
最近在写模组的时候发现渲染器里面离不开模型的定义,在渲染的第二篇文章中简单的讲解了一下关于模型部分的内容,其实不管是方块还是方块实体,都离不开模型的内容 🧱 一、CubeListBuilder 功能解析 CubeListBuilder 是 Minecraft Java 版模型系统的核心构建器,用于动态创…...
vue3+vite项目中使用.env文件环境变量方法
vue3vite项目中使用.env文件环境变量方法 .env文件作用命名规则常用的配置项示例使用方法注意事项在vite.config.js文件中读取环境变量方法 .env文件作用 .env 文件用于定义环境变量,这些变量可以在项目中通过 import.meta.env 进行访问。Vite 会自动加载这些环境变…...
Yolov8 目标检测蒸馏学习记录
yolov8系列模型蒸馏基本流程,代码下载:这里本人提交了一个demo:djdll/Yolov8_Distillation: Yolov8轻量化_蒸馏代码实现 在轻量化模型设计中,**知识蒸馏(Knowledge Distillation)**被广泛应用,作为提升模型…...
return this;返回的是谁
一个审批系统的示例来演示责任链模式的实现。假设公司需要处理不同金额的采购申请,不同级别的经理有不同的审批权限: // 抽象处理者:审批者 abstract class Approver {protected Approver successor; // 下一个处理者// 设置下一个处理者pub…...
