把Scheme翻译成Java和C++的工具
一、为什么要写这个工具?
公司内容有多个项目需要同一个功能,而这些项目中,有的是用Java的,有的是用C++的,同时由于某些现实条件限制,无法所有项目都调用统一的服务接口(如:可能运行在无网络的情况下),所以可选方案有两种:
a.分别用Java和C++实现两套一样的功能。
b.实现一个生成工具,用来生成同一种逻辑的Java和C++代码。
现在项目还刚刚开始,还不确定未来会使用以上两种方案之中的哪一种,不过我还是需要事先做准备。
二、本文不会有的内容
我不希望在一篇技术文章中涉及到任何业务,以及为避免任何形式的公司敏感信息的泄露,所以本文中不会出现这个工具中的关键代码,即:
a.这篇文章中的代码都是用于描述方便而写的测试代码,而不会包含真实的项目代码。
b.这篇文章只讲技术,而不涉及业务处理。
三、为什么用Scheme来做为源语言?
首先要说一下,这个源语言是一个Scheme-like的语言,而不是Scheme本身,但语法非常相似。
那么为什么用Scheme的形式来做为源语言呢?直接用Java不就只需要把Java翻译成C++了么?
我选择用Scheme的原因是:
a.这个语言是一个动态类型语言,所以在语法上不会在声明中出现对象的类型。
b.Scheme-like实际上就是lisp-like(S表达式),这种语法形式非常简单,而且各种语言元素上高度统一,非常容易实现。
基于以上两点,以后如果我们再写一个图形工具(给业务方用户直接使用来编辑算法)来生成这个语言也会比较容易——如果要写一个GUI工具编辑好算法之后,生成的直接是Java代码,则有点过于困难了,虽然还是可以做到,但工作量上可能会是10倍,甚至数十倍的差距。所以,为了未来可能的处理方便,以及现在对语言本身的实现方便,这里选用一个在语言上“看似”无类型的,且语法简单的语法来做是比较好的。同时,因这个语言整个就是我在做,所以以后在现在语法的基础之上增加更具表达力的语法也并不会再困难。
四、关于类型
Scheme语言是一个动态类型的语言,而我的目标语言是Java和C++两个静态类型语言,所以这个Scheme-like语言只是在形式上是动态语言的,而本制上必须还是一个静态类型语言才行。那么即然在语言的文本上并不存在关于类型的声明,那么我们怎么知道翻译到Java和C++之后是什么类型呢?
关于类型识别分为如下几种情况:
a.业务代码直接掉用的接口是最简单的……后续补充。
b.函数内部的变量定义……后续补充。
c.对于内部的一个函数,情况最是复杂,比如:(define (fun m a) (m a))这样一个函数可以看出m是一个函数(或一个lambda),它接受一个参数a,但我们从这个函数中无法确定a是什么类型的,m的返回值是什么。而这此类型在Scheme中也是只有当运行时才可以确定的,但我们需要直接翻译到静态的代码中,这样我们实际上不得不通过一些手段以确定期类型……后续补充。
d.……
五、表达式与函数
在Scheme中,表达式都是有值的,所以我们可以写出这样的代码:(let ((a (if (< 1 2) 1 2))) a),而在C++和Java中表达式都是没有值的,所以我们不可以这样写:double x = {if (true) 1; else 2;},这样在本身的语法映射上就差别太大了。为了简化开发,还是把Scheme中的函数,都对应的在Java和C++代码中实现一个函数,这样就会比罗容易做这个翻译。
比如:在Scheme中if是一个函数,则在Java中实现一个这样的接口<T> T _if(Supplier<Boolean> cond, Supplier<T> exeBody1, Supplier<T> exeBody2);在C++中实现这样一个函数template<class T> T _if(std::function<bool()> cond, std::function<T()> exeBody1, std::function<T()> exeBody2);——这里使用lambda式参数的原因会在后面讲到。
再比如:在Scheme中+是一个函数,则在Java中实现一个这样的接口double add(double d1, double d2),在C++中也实现同样一个函数。
等等。
这些函数的实现都是只有一行的,C++编译器都是可以用inline的方式优化掉的,所以不会有什么性能问题,而Java语言并不会做inline这样的动作,但我就不清楚JVM是否会在运行时进行inline处理了。
六、关于延时计算
延时计算主要应用在两个方面:
……
七、关于短路
……
今天有点累了,改天再补完吧。
把Scheme翻译成Java和C++的工具的更多相关文章
- JavaScript翻译成Java
这两天公司有一个需求,将一段加密的JavaScript代码转换为JAVA版. JavaScript中的某一段代码: 前期查看了整个JavaScript代码,发现代码中,方法里面嵌套方法,各种不合规的变 ...
- 一脸懵逼学习Hive的安装(将sql语句翻译成MapReduce程序的一个工具)
Hive只在一个节点上安装即可: 1.上传tar包:这个上传就不贴图了,贴一下上传后的,看一下虚拟机吧: 2.解压操作: [root@slaver3 hadoop]# tar -zxvf hive-0 ...
- 【Java】 剑指offer(46) 把数字翻译成字符串
本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集 题目 给定一个数字,我们按照如下规则把它翻译为字符串:0翻译成" ...
- java中为什么把Checked Exception翻译成受检的异常?
6.Checked Exception(受检的异常) 马克-to-win:为什么我大胆的把Checked Exception翻译成受检的异常?因为这类异常,编译器检查发现到它后会强令你catch它或t ...
- 关于 调用 JNI JAR 的说明和注意事项,调用第三方 JAR SDK 和 翻译 安卓 JAVA 代码 的说明 V2015.6.10
关于 调用 JNI JAR 的说明和注意事项,调用第三方 JAR SDK 和 翻译 安卓 JAVA 代码 的说明 V2015.6.10 转载请标明出处,否则死全家.选择[复制链接]即可得到出处. (* ...
- 【Offer】[46] 【把数字翻译成字符串】
题目描述 思路分析 测试用例 Java代码 代码链接 题目描述 给定一个数字,我们按照如下规则把它翻译为字符串: 0翻译成"a",1翻译成"b",..... 1 ...
- ANTLR4将BF翻译成CPP
实验环境: 操作系统:windows 10 JAVA:JDK 1.8 antlr:antlr-4.7.1-complete.jar IDE:IntelliJ IDEA 2017.2.7 实验目的: 实 ...
- 剑指 Offer 46. 把数字翻译成字符串 + 动态规划
剑指 Offer 46. 把数字翻译成字符串 Offer_46 题目描述 题解分析 本题的解题思路是使用动态规划,首先得出递推公式如下 dp[i] = dp[i-1]+dp[i-2](如果s[i-1] ...
- 【探索】机器指令翻译成 JavaScript
前言 前些时候研究脚本混淆时,打算先学一些「程序流程」相关的概念.为了不因太枯燥而放弃,决定想一个有趣的案例,可以边探索边学. 于是想了一个话题:尝试将机器指令 1:1 翻译 成 JavaScript ...
随机推荐
- 配置server禁止全部非法域名 訪问自己的server
1.Apache2.4.1曾经: 第一种 直接拒绝訪问 打开 httpd.conf 文件,将一下配置追加到文件最后. #直接拒绝全部非法域名 <VirtualHost *:80> Ser ...
- [办公应用]让WORD自动显示到四级目录
一般情况下,word的目录默认显示到三级目录.如果需要显示到四级目录,你会怎么操作呢? 只要按下图所示,单击“插入”-“引用”-“索引和目录” 然后单击“目录”选项卡,将“显示级别”处的3改为4即可. ...
- 局域网内PC通过笔记本共享上网
现实:PC.笔记本都通过网线接在局域网内,局域网无法上网:笔记本有无线网卡,可连WIFI上网. 现在想让PC通过笔记本来共享上网. 步骤: 1.笔记本开启DHCP.方法是开启"服务" ...
- Hibernate 之 二级缓存
在上篇文章中我们对缓存以及Hibernate的一级缓存进行了介绍,接下来的主要内容将是Hibernate的二级缓存. 二级缓存也称为进程级的缓存或SessionFactory级的缓存,二级缓存可以被所 ...
- Ubuntu 12.10终端Terminal快捷方式调用
1:使用快捷键:ctrl+alt+t 打开终端 2:在终端上右键,选“Lock to launcher” 这样就锁定在左侧了,需要用时,直接点就打开了.
- facebook Presto SQL分析引擎——本质上和spark无异,分解stage,task,MR计算
Presto 是由 Facebook 开源的大数据分布式 SQL 查询引擎,适用于交互式分析查询,可支持众多的数据源,包括 HDFS,RDBMS,KAFKA 等,而且提供了非常友好的接口开发数据源连接 ...
- 运算符:三目运算符,运算符优先级,sizeof,自增自减,取余
一://---------运算符-----------// 1.运算符是告诉编译程序执行特定算术或逻辑操作的符号. 2.按照功能划分: 算术运算符. 关系运算符与逻辑运算符.按位运算符. 3.运算符根 ...
- bzoj1047 [HAOI2007]理想的正方形——二维单调队列
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1047 就是先对行做一遍单调队列,再对那个结果按列做一遍单调队列即可. 代码如下: #incl ...
- 前端之html第二天
一.内容
- vue 基础知识随笔
在vue2.0中一个vue实例的生命周期中已经没有ready()了,在vue1.0中才有ready();在vue2.0中立即执行函数使用mounted v-for 参数顺序更新: 数组中使用(valu ...