java开发C语言解释器:结构体内存与成员变量的一致性

2024-04-30 22:32

本文主要是介绍java开发C语言解释器:结构体内存与成员变量的一致性,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

更详细的讲解和代码调试演示过程,请参看视频
用java开发C语言编译器

如果你对机器学习感兴趣,请参看一下链接:
机器学习:神经网络导论

由于设计上的原因,我们的解释器在解读结构体指针时,存有两套信息存储系统,例如对于结构体定义和结构体指针变量的相关代码如下:

struct TAG {int  p;char c;int arr[3];}tag;struct TAG *pTag;
int sz = sizeof(tag);
pTag = malloc(sz);

我们根据结构体TAG 定义了一个结构体指针pTag, 并且通过malloc为该指针分配了动态内存,在解释器内部,由于设计上的原因,一个结构体指针变量应对着两套内存系统,一个是解释器会为结构体指针遍历中的每一个成员变量创建一个Symbol对象,这些对象存储在符号表中,二是解释器根据malloc调用,为该指针分配了动态内存,具体情况如下图所示:
这里写图片描述

这样带来一个问题是,我们必须保证两套内存数据的值必须是一致的,例如成员变量p的值改变了后,那么指针指向的动态内存,头四个字节的内容也需要相应改变。上一节,我们实现的就是这个功能。

本节,我们要实现的是,如果代码修改了动态内存头四个字节的数据,那么结构体的成员变量p的值也需要跟着改变。上一节做法的思路是,解释器监视结构体成员变量,每当结构体成员的值发生改变,那么解释器就把所有成员变量的值复制到内存中。这次,我们反其道而行之,在解释器读取结构体成本变量的数值之前,先把内存中的信息拷贝到每个结构体成员中。

完成本节代码后,解释器将能正确的解释执行如下代码:

void main() {struct TAG {char c;char b[3];int p;}tag;struct TAG* pTag;int sz ;sz = sizeof(tag);pTag = malloc(sz);pTag[0] = 3;    pTag[1] = 4;pTag[2] = 5;pTag[3] = 6;printf("value of c is %d, b[0] is %d, b[1] is %d, b[2] is %d ", pTag->c, pTag->b[0], pTag->b[1], pTag->b[2]);
}

当结构体成员变量被赋值或读取的语句是 pTag->c, 此类语句对应的语法表达式为:Unary -> Unary StructOP Name

负责解释执行该类语句的解释器代码处于UnaryNodeExecutor.java中,相关代码逻辑如下:

public class UnaryNodeExecutor extends BaseExecutor implements IExecutorReceiver{private Symbol structObjSymbol = null;private Symbol monitorSymbol = null;@Overridepublic Object Execute(ICodeNode root) {executeChildren(root);int production = (Integer)root.getAttribute(ICodeKey.PRODUCTION); String text ;Symbol symbol;Object value;ICodeNode child;switch (production) {....case CGrammarInitializer.Unary_StructOP_Name_TO_Unary:child = root.getChildren().get(0);String fieldName = (String)root.getAttribute(ICodeKey.TEXT);symbol = (Symbol)child.getAttribute(ICodeKey.SYMBOL);if (isSymbolStructPointer(symbol)) {copyBetweenStructAndMem(symbol, false);}Symbol args = symbol.getArgList();while (args != null) {if (args.getName().equals(fieldName)) {break;}args = args.getNextSymbol();}if (args == null) {System.err.println("access a filed not in struct object!");System.exit(1);}root.setAttribute(ICodeKey.SYMBOL, args);root.setAttribute(ICodeKey.VALUE, args.getValue());if (isSymbolStructPointer(symbol) == true) {checkValidPointer(symbol);structObjSymbol = symbol;monitorSymbol = args;ExecutorBrocasterImpl.getInstance().registerReceiverForAfterExe(this);} else {structObjSymbol = null;}break;....}
....
}

在上面代码中,我添加了如下语句:

if (isSymbolStructPointer(symbol)) {copyBetweenStructAndMem(symbol, false);}

这几行代码的作用是,判断当前Symbol所对应的变量是否是结构体指针,如果是的话,在访问该结构体变量的成员之前,先把数据从动态内存,也就是上图中Mem所代表的那部分,拷贝到各个结构体成员变量中。我们再看看copyBetweenStructAndMem 接口的实现:

private void copyBetweenStructAndMem(Symbol symbol, boolean isFromStructToMem) {Integer addr = (Integer)symbol.getValue();MemoryHeap memHeap = MemoryHeap.getInstance();Map.Entry<Integer, byte[]> entry = memHeap.getMem(addr);byte[] mems = entry.getValue();Stack<Symbol> stack = reverseStructSymbolList(symbol);int offset = 0;while (stack.empty() != true) {Symbol sym = stack.pop(); try {if (isFromStructToMem == true) {offset += writeStructVariablesToMem(sym, mems, offset); } else {offset += writeMemToStructVariables(sym, mems, offset);}} catch (Exception e) {System.err.println("error when copyin struct variables to memory");e.printStackTrace();}}}

它输入两个参数,第一个是结构体指针变量对应的Symbol对象,第二个参数是一个boolean 类型变量,如果他的值是true, 那么该函数调用writeStructVariablesToMem 将结构体成员变量的值拷贝到内存,如果是false, 那么调用writeMemToStructVariables 把内存数据拷贝到结构体成员变量中。

writeStructVariablesToMem 的逻辑我们上节讲过了,本节我们看看writeMemToStructVariables的实现:

private int writeMemToStructVariables(Symbol symbol, byte[] mem, int offset) throws Exception {if (symbol.getArgList() != null) {//struct variable, copy mem to struct recursivelyreturn writeMemToStructVariables(symbol, mem, offset);}int sz = symbol.getByteSize();int val = 0;if (symbol.getDeclarator(Declarator.ARRAY) == null) {val = fromByteArrayToInteger(mem, offset, sz);symbol.setValue(val);} else {return copyMemToArrayVariable(symbol, mem, offset);}return sz;}private int fromByteArrayToInteger(byte[] mem, int offset, int sz) {int val = 0;switch (sz) {case 1:val = mem[offset];break;case 2:val = (mem[offset + 1] << 8 | mem[offset]);break;case 4:val = (mem[offset+3] << 24 | mem[offset+2] << 16 | mem[offset + 1] << 8 | mem[offset]);break;}return val;}private int copyMemToArrayVariable(Symbol symbol, byte[] mem, int offset) {int sz = symbol.getByteSize();Declarator declarator = symbol.getDeclarator(Declarator.ARRAY);if (declarator == null) {return 0;}int size = 0;int elemCount = declarator.getElementNum();for (int i = 0; i < elemCount; i++) {int val = fromByteArrayToInteger(mem, offset + size, sz);size += sz;try {declarator.addElement(i, val);} catch (Exception e) {// TODO Auto-generated catch blocke.printStackTrace();}}return size;}

writeMemToStructVariables 实现的基本逻辑是,根据结构体成员变量在内存中的相对位置,以及该变量对应的字节大小,从内存中读取相关数据,把读取的信息写入到该变量所对应的Symbol对象中。例如根据给定的代码实例,结构体的成员变量c,位于结构体偏移0处,它是char类型,因此该变量对应的字节大小为1,于是,我们从动态内存其实地址为0开始,读取一个字节的数据内容,把读取的内容设置到变量c在符号表中的Symbol对象。

fromByteArrayToInteger的作用是把多个字节的数据合成一个基本数据类型,例如int类型的变量是4个字节,那么该函数先从内存数组里读出4个字节的数据,然后把这4个字节合并成一个int整形数。

copyMemToArrayVariable的作用跟我们上节讲过的函数copyArrayVariableToMem, 逻辑是一样的。

完成上面代码后,当我们通过指针修改了结构体指针指向的内存后,修改的结果就能直接反应到结构体的成员变量上,我们给定的示例代码,解释器解释执行后结果如下:
value of c is 3, b[0] is 4, b[1] is 5, b[2] is 6

由此可见,我们解释器对源代码的解释执行结果是正确的。请参看视频以便获得更详细的讲解和更详实的代码调试演示过程。

更多技术信息,包括操作系统,编译器,面试算法,机器学习,人工智能,请关照我的公众号:
这里写图片描述

这篇关于java开发C语言解释器:结构体内存与成员变量的一致性的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/950079

相关文章

SpringBoot整合liteflow的详细过程

《SpringBoot整合liteflow的详细过程》:本文主要介绍SpringBoot整合liteflow的详细过程,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋...  liteflow 是什么? 能做什么?总之一句话:能帮你规范写代码逻辑 ,编排并解耦业务逻辑,代码

JavaSE正则表达式用法总结大全

《JavaSE正则表达式用法总结大全》正则表达式就是由一些特定的字符组成,代表的是一个规则,:本文主要介绍JavaSE正则表达式用法的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录常用的正则表达式匹配符正则表China编程达式常用的类Pattern类Matcher类PatternSynta

Spring Security中用户名和密码的验证完整流程

《SpringSecurity中用户名和密码的验证完整流程》本文给大家介绍SpringSecurity中用户名和密码的验证完整流程,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定... 首先创建了一个UsernamePasswordAuthenticationTChina编程oken对象,这是S

java实现docker镜像上传到harbor仓库的方式

《java实现docker镜像上传到harbor仓库的方式》:本文主要介绍java实现docker镜像上传到harbor仓库的方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 前 言2. 编写工具类2.1 引入依赖包2.2 使用当前服务器的docker环境推送镜像2.2

Go语言中nil判断的注意事项(最新推荐)

《Go语言中nil判断的注意事项(最新推荐)》本文给大家介绍Go语言中nil判断的注意事项,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1.接口变量的特殊行为2.nil的合法类型3.nil值的实用行为4.自定义类型与nil5.反射判断nil6.函数返回的

Java easyExcel实现导入多sheet的Excel

《JavaeasyExcel实现导入多sheet的Excel》这篇文章主要为大家详细介绍了如何使用JavaeasyExcel实现导入多sheet的Excel,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录1.官网2.Excel样式3.代码1.官网easyExcel官网2.Excel样式3.代码

Java MQTT实战应用

《JavaMQTT实战应用》本文详解MQTT协议,涵盖其发布/订阅机制、低功耗高效特性、三种服务质量等级(QoS0/1/2),以及客户端、代理、主题的核心概念,最后提供Linux部署教程、Sprin... 目录一、MQTT协议二、MQTT优点三、三种服务质量等级四、客户端、代理、主题1. 客户端(Clien

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

Spring 框架之Springfox使用详解

《Spring框架之Springfox使用详解》Springfox是Spring框架的API文档工具,集成Swagger规范,自动生成文档并支持多语言/版本,模块化设计便于扩展,但存在版本兼容性、性... 目录核心功能工作原理模块化设计使用示例注意事项优缺点优点缺点总结适用场景建议总结Springfox 是