body, table{font-family: 微软雅黑; font-size: 13.5pt}
table{border-collapse: collapse; border: solid gray; border-width: 2px 0 2px 0;}
th{border: 1px solid gray; padding: 4px; background-color: #DDD;}
td{border: 1px solid gray; padding: 4px;}
tr:nth-child(2n){background-color: #f8f8f8;}

   地址解释程序是一些 CODE 词,计算机类型不同,这些 CODE 词也不同,故地址解释程序又往往称为内部解释程序。相应地,又把文本解释程序称为外部解释程序。
   IP  解释指针       W  现行词指针 。 IP 通常指着在一个冒号定义中的下一个要被执行的词,而 W 则指着当前要执行的词(W 的内容为当前要执行的词的 cfa);

地址解释程序:【
      在文本解释程序中通过 EXECUTE 调用内部解释程序。EXECUTE 初始化地址解释程序,它接过留在堆栈上的代码指针域地址 cfa ,然后间接转移执行由此地址所指的子程序。
CODE   EXECUTE    ( cfa -  )              执行编译地址(即 cfa )在参数栈顶的定义                   
       W   POP             把在堆栈上的 cfa 送入到现行词指针 W 中
       0   [W]   JMP      经由 W 进行一次间接转移
END-CODE
内部解释程序由定义 NEXT、NEST 及 UNNEST所构成。他们是由主机的汇编语言或宏汇编语言编写的。
NEXT   是一个 CODE 词。执行 NEXT 就是执行由 IP 所指着的下一个词。在执行 NEXT 中要做三件事:
1、((IP)) -> W   使 W 指着由 IP 当前内容所指着的词;
2、(IP)+2 -> IP   使 IP 的内容递增 2;
3、间接转移执行由 W 所指着的词;


解释指针 IP (有时又称为指令指针)的值由 NEXT 来设置和更新,所以每一个 CODE 词的最后都无例外地一定要执行 NEXT 。执行 NEXT 就去执行由 IP 所指的下一个 cfa 所代表的子程序,同时更新 IP 的值指向再下一个地址,一直重复直到执行完毕一个定义中的所有编译地址。
NEST    在执行 NEST 之前,IP 含有待返回的地址,而 W 含有要被调用的冒号定义的 cfa 。执行 NEST 要做四件事:
1、(IP) -> 返回栈顶(即保存好的 IP 原来的内容);
2、(W)+2 -> W   使 W 指向要被调用的定义的 pfa (即参数域地址);
3、(W) -> IP   把所要执行的词的地址送入 IP;
4、执行 NEXT。


NEST 在不同的 FORTH 系统中可能有不同的名字,如(NEST)、DO_COLON等。NEST 所代表的子程序是所有的冒号定义在一开始执行时都要首先执行的共同子程序。
UNNEST   是一个 CODE 词。它把返回栈顶的首项移入到解释指针中,结束一冒号定义的执行和返回到它的调用者。执行 UNNEST 时要做两件事:
1、把返回堆栈顶第一个单元的内容送入 IP 中;
2、执行 NEXT ;


UNNEST 在不同的 FORTH 版本中也可能有不同的名字,比如 ;S 、EXIT 等。在词典中的每一个冒号定义的最后一个成分一定是 UNNEST ,它来源于冒号源定义中最后的分号。


冒号定义的执行过程:
eg:
:  NOTHING   ;
:  STEP1   DUP   OVER   NOTHIN  +  +  ;   // 用词 ' 可以求出每个词的 cfa 地址;
NOTHING ^nest ^unnest(14B)
66DE 66E0 ( lfa ) 66E8 66EA ( pfa )
STEP1 ^NEST ^DUP(3C4) ^OVER(3E4) ^NOTHING(66E8) ^+(582) ^+(582) ^UNNEST(14B)
66F2 66F6 66F8 66FA 66FC 66FE 6700 6702
当执行 DUP 时,(IP)=66FA指向OVER,DUP词本身的最后一个成分是NEXT,于是当执行DUP的NEXT时发生:
1、((IP)) -> W ,也即IP的内容的内容进入到 W 中,于是 (W)=3E4,它是词 OVER 的 cfa;
2、(IP)+2 -> IP,于是 (IP)=66FC,指向词NOTHING;
3、执行 W 所指的词,即执行 OVER 。OVER 是一个 CODE 词,它的 cfa 单元的内容就是它的 pfa ,于是就跳去执行其词身内的机器指令。
执行 OVER 的最后一个成分又是 NEXT,当执行此 NEXT 时发生:
1、((IP)) -> W,于是 (W) = 66E8,它是 NOTHING 的 cfa;
2、(IP)+2 -> IP,于是 (IP) = 66FE,它指着第一个 +;
3、执行 W 所指的词 NOTHING;
NOTHING 是以冒号定义,所以它的 cfa 单元的内容是 NEST 的地址,于是就跳去执行 NEST,执行 NEST 时发生:
1、(IP) -> 返回堆栈栈顶,返回地址 66FE 就被保存在返回栈中。
2、(W)+2 -> W,(W)=66EA,它是 NOTHING 的 pfa。
3、(W) -> IP,(IP)=66EA,指着 NOTHING 内的 UNNEST。
4、执行 NEXT


1)、((IP)) -> W,(W)=14B,它是 UNNEST 的地址。
2)、(IP)+2 -> IP,(IP)=66EC,这个地址是分号后的一个地址,没有任何意义,后面执行UNNEST会被修改,直接丢弃
3)、执行W所指的词,也即执行UNNEST
执行 UNNEST 时发生:
1、返回栈顶的内容 -> IP,于是(IP)=66FE,指着 STEP1 内的第一个 +;    //每执行一次UNNEST,程序执行嵌套就减少一层,返回到原来的调用层中
2、执行 NEXT
1)、((IP)) -> W,(W) = 582
2)、(IP)+2 -> IP,(IP) = 6700,指着 STEP1 中的第二个 +;
2)、执行由W所指的词,也即此时 STEP1 中的第一个 + 被执行。                 //由于 + 是CODE词,它不会进入更深一层的嵌套。
……等执行完STEP1的UNNEST之后就返回到STEP1的调用者,也就是返回到外部解释程序INTERPRET,由于执行完STEP1后输入流耗尽,于是跳转到FORTH循环中的外循环 QUIT 。在显示屏上打印出“ok”后,FORTH 又等待使用者从键盘上输入下一行命令。

Forth 内部解释程序工作流程的更多相关文章

  1. Spark基本工作流程及YARN cluster模式原理(读书笔记)

    Spark基本工作流程及YARN cluster模式原理 转载请注明出处:http://www.cnblogs.com/BYRans/ Spark基本工作流程 相关术语解释 Spark应用程序相关的几 ...

  2. 前端工作流程自动化——Grunt/Gulp 自动化

    什么是自动化 先来说说为什么要自动化.凡是要考虑到自动化时,你所做的工作必然是存在很多重复乏味的劳作,很有必要通过程序来完成这些任务.这样一来就可以解放生产力,将更多的精力和时间投入到更多有意义的事情 ...

  3. Struts2 工作流程

    Struts2使用了WebWork的设计核心(XWork),在内部使用拦截器处理用户请求,从而允许用户业务逻辑控制器和ServletAPI分离.Struts2内部是一个MVC架构,Struts2 的核 ...

  4. docker工作流程

    Docker提供一种方法在容器中运行安全隔离的应用程序,应用程序与所有依赖项和库一起打包在容器中.因为你的应用程序总是可以使用它在构建镜像中期望的环境运行,测试和部署比以往任何时候都更简单,因为你的构 ...

  5. SpringMVC第一篇【介绍、入门、工作流程、控制器】

    什么是SpringMVC? SpringMVC是Spring家族的一员,Spring是将现在开发中流行的组件进行组合而成的一个框架!它用在基于MVC的表现层开发,类似于struts2框架 为什么要使用 ...

  6. Struts2学习---拦截器+struts的工作流程+struts声明式异常处理

    这一节我们来看看拦截器,在讲这个之前我是准备先看struts的声明式异常处理的,但是我发现这个声明式异常处理就是由拦截器实现的,所以就将拦截器的内容放到了前面. 这一节的内容是这样的: 拦截器的介绍 ...

  7. 【嵌入式开发】 Bootloader 详解 ( 代码环境 | ARM 启动流程 | uboot 工作流程 | 架构设计)

    作者 : 韩曙亮 博客地址 : http://blog.csdn.net/shulianghan/article/details/42462795 转载请著名出处 相关资源下载 :  -- u-boo ...

  8. [Inside HotSpot] C1编译器工作流程及中间表示

    1. C1编译器线程 C1编译器(aka Client Compiler)的代码位于hotspot\share\c1.C1编译线程(C1 CompilerThread)会阻塞在任务队列,当发现队列有编 ...

  9. kafka工作流程| 命令行操作

    1.  概述 数据层:结构化数据+非结构化数据+日志信息(大部分为结构化) 传输层:flume(采集日志--->存储性框架(如HDFS.kafka.Hive.Hbase))+sqoop(关系型数 ...

随机推荐

  1. archer docker安装部署

    1.准备配置文件从archer项目官网下载/archer/settings.py文件,根据自己情况放到相应的目录我下载后放到如下目录[root@lenovo opt]# mkdir -p /opt/a ...

  2. 原生JS封装创建多级菜单函数

    手写一个使用原生JS封装的多级菜单的函数,满足以下几点需求. 子类层级不确定,可根据数据自动生成多级菜单. 操作便捷,只需传入一个HTML标签. 缺点: 需要满足特定的数据结构 废话不多说,展示代码. ...

  3. xml转json和实体类的两种方式

    本文为博主原创,未经允许不得转载: xml在http通信中具有较高的安全性和传输速度,所以应用比较广泛, 在项目中往往需要对xml,json和实体类进行相互转换,在这里总结一下自己所用到的一些方法: ...

  4. linux中查找(find、locate、which、whereis、grep)命令汇总、帮助命令(man、whatis、apropos、info、help)汇总

    (一)find命令详解 find:功能:文件搜索: 语法:find[搜索范围][匹配条件]:  只要匹配条件完全符合才可以被显示,使用通配符*匹配条件*则可以显示匹配条件的所有目录,问号?匹配单个字符 ...

  5. testng.xml中groups标签使用

    XML配置如下: <?xml version="1.0" encoding="UTF-8"?> <suite name="suite ...

  6. vue-router 中踏过的坑

    1.做完页面滚动,然后再加上路由,发现路由一直跳转不了,经历千辛万苦才发现是BScroll没有配置click:true,当看过文档时心里一万只草泥马奔腾而过,我预感到成长道路上还有多少坑在等着我. 2 ...

  7. linux 修改普通用户的 max user process

    因为出现  fork: retry: No child processes 问题 , google了一下 , 大家说是要去修改 /etc/security/limits.conf 文件 , 然后我用r ...

  8. python常用技巧

    1,关于tab键与4个空格: 由于不同平台间,tab键值设置有所区别,据相关介绍,官方在缩进方面推荐使用4个空格.方便起见,可设置tab自动转换为4个空格. 1.1在pycharm中:    通过fi ...

  9. pom.xml配置,针对mvn clean install -P参数(环境参数)打包

    pom.xml配置,针对mvn clean install -P参数(环境参数)打包 比如你有2个环境,一个dev,一个prod, 然后你在mvn打包的时候,可以通过-P来打包,是打dev包,还是pr ...

  10. textarea跟随内容自动伸缩高度实现方案

    监听input事件,然后将textarea的style.height设置为最低高度(19px),进而获取到元素的scrollHeight,然后将scroolHeight设置为style.height