轻量级解释型弱类型语言 (迭代中…)

这是我基于 Golang 开发的一个轻量级解释型语言,目前仍在持续开发中。还没有正式的名字,暂且叫它 Lit,容易记。原本我只打算用它去解析文本中的算术表达式,后来逐步地扩充了更多的特性。索性将它定位为一个解释型编程语言了。

目前我已经实现了一些编程语言必备的基础功能,尤其是比较重要的算术表达式、弱类型转换、内置函数、变量声明、函数、if语句等。仍然需要进一步完善。这里面许多的特性参考了 Go、PHP 以及 Node.js。

最终我要将它实现为一个解释型的弱类型编程语言,希望它能用于日常的 Web 开发。至于能否顺利完成,还需要很长时间才能下定论。


现在已支持的特性

一、变量 (声明、赋值、拼接、参与算术计算)

二、完整的算术表达式 [+ - * / & | ^ %]

三、弱类型

四、关系运算符 [> < >= <= != == ===]

五、逻辑运算符 [&& ||]

六、内置函数 (试验阶段)

七、自定义函数

八、if 语句 (if, else if, else) 试验阶段


计划在后期支持的特性

一、数组

二、for

三、对象或结构体 (从中选择其一实现 会参考golang)

四、递归

五、基础的动态库 (如 tcp/http/mysql等)

六、异常处理


使用方法 非常简单 直接引入就行了

go get github.com/pywee/lit

一、变量声明

import "github.com/pywee/lit"

func main() {
    // 执行 exprs 内的代码 将输出 580
    exprs := `
        a = 123;
        b = a + 456;
        print(b + 1); // 580
        `
    _, err := lit.NewExpr([]byte(exprs))
}


二、算术表达式的计算。算术符号的优先级保持与 Go 语言相同。示例:

// 不同的语言符号优先级是不完全一样的,Lit 的算术符号优先级保持与 Golang 一致
// 首先我们执行 Go 语言原生函数进行数学表达式计算
// 以下句子最终会输出 
// +1.600000e+001
// 125
println((2 + 100 ^ 2 - (10*1.1 - 22 + (22 | 11))) / 10 * 2)
println(12/333+31+(5/10)-6|100)

// 使用 Lit 计算文本中的数据
// 表达式文本
// 执行下面的句子 最终会输出 16 125
exprs := `
    a = (2 + 100 ^ 2 - (10*1.1 - 22 + (22 | 11))) / 10 * 2;
    b = 12 / 333 + 31 + (5 / 10) - 6 | 100;
    print(a); // 16
    print(b); // 125
`
_, err := lit.NewExpr([]byte(exprs))

// 同样的表达式放在 PHP 中,会输出 -24


三、自定义函数及内置函数


// 下面的句子调用了两个函数 
// isInt(arg) 用来检查 arg 是否为整型 
// replace(arg1, arg2, arg3, arg4) 用来做字符串替换
// 最终会输出
// STRING hello word
exprs := `
    a = replace("hello word111", "1", "", 2-isInt((1+(1 + isInt(123+(1+2)))-1)+2)-2);
    varDump(a);
`
_, err = lit.NewExpr([]byte(exprs))


// 执行下面语句 最终会输出 100
exprs := `
    func demo() {
        return 1-false*3/10+(20-1);
    }
    print(demo()+80); // 100
`
_, err = lit.NewExpr([]byte(exprs))


四、弱类型转换。弱类型的特性我把它设计为与 PHP 基本一样

// 当布尔值参与运算时,底层会将 true 转为 1, false 转为 0
// 执行下面句子 将输出
// 0
// true
// true
// true
// true
exprs := `
    a = true - 1;
    b = isInt(1);
    c = isFloat(1.0);
    d = false == 0.0;
    e = "false" == 0.0;
    print(a); // 0
    print(b); // true
    print(c); // true
    print(d); // true
    print(e); // true
`
_, err := lit.NewExpr([]byte(exprs))


exprs := `
    a = false;
    b = true;
    c = "123";
    d = 456;
    print(a + b); // 1
    print(a >= b); // false
    print(a + b + c + d); // 580
`
_, err := lit.NewExpr([]byte(exprs))

与其他弱类型语言一样,字符串数字与整型相操作,底层会将字符串数字转换为整型
执行下面句子 将输出 0 和 true


src := `
    a = "1" - 1;
    b =  0.0 >= false+1 || (1<=21 && 1==1);
    print(a); // 0
    print(b); // true
`
_, err := lit.NewExpr([]byte(src))

与其他弱类型语言一样,字符串数字与整型相操作,底层会将字符串数字转换为整型
执行下面句子将输出 0

src := `
    a = "1" - 1;
    print(a); // 0
`
_, err := lit.NewExpr([]byte(exprs))

字符串与字符串相加时 将进行字符串的拼接
执行以下句子,将会输出 abcdef

src := `
    a = "abc" + "def";
    print(a); // abcdef
`
_, err := lit.NewExpr([]byte(src))

但如果当两个字符串都为数字时 对他们进行相加 则会被底层转换为数字
执行如下句子,将会输出 579

src := `
    a = "123" + "456";
    print(a); // 579
`
_, err := lit.NewExpr([]byte(src))

其他字符串+整型将会报错
执行如下句子会报错

src := `
    a = "abcwwww1230"+0.01;
    print(a); // 报错
`
_, err := lit.NewExpr([]byte(src))

// 在 php 里面,执行上面的句子不但不会报错,还会将字符串 "abcwwww1230" 中的 1230 单独提取出来
// 并且还会将 1230 转换成整型与后面的 0.01 进行计算
// 我不太清楚支持这种操作的目的是什么,个人感觉,这显然没有一点点运用场景
// 所以在 Lit 里面 不允许这样操作 既损耗效率又没有一点点意义


五、“并且” 与 “或者” 符号处理

执行如下句子,将会输出 BOOL true

src := `
    a = isInt(1) && 72+(11-2) || 1-false;
    varDump(a); // BOOL true
`
_, err := lit.NewExpr([]byte(src))


六、if语句 (试验阶段)

执行如下句子,将会输出 20

src := `
func demo() {
    if 1+1 == 5 {
        return 12345;
    } else if 3 == 2 {
        return 3
    }
    return 1-false*3/10+(20-1);
}
print(demo()); // 20
`
_, err := lit.NewExpr([]byte(src))


请注意,Lit 的算术符号优先级向 Golang 看齐。每个语言对算术符号的优先级处理都有一定区别,如,针对以下表达式进行计算时:

// 2 + 100 ^ 2 - (10*1.1 - 22 + (22 | 11)) / 10 * 2
// PHP 输出 -104
// Node.js 输出 -104
// Golang 输出 96
// Lit 输出 96

Lit 算术符号优先级

() && || 
> < >= <= == != ===
* / % 
| &
+ - ^ 

当前支持的内置函数有如下,更多函数将会在逐步补充 (当前仍然存在bug)
通用处理函数

    print
    varDump

字符串处理函数
函数的命名基本参考了 Go 语言
除了个别函数有差别,如
utf8Len 用于检测字符串字数的函数
isNumeric 用于判断当前输入是否为数字

trim
trimLeft
trimRight
trimSpace
len
utf8Len
md5
replace
contains
index
lastIndex
toLower
toUpper
toTitle
repeat

其他函数

isNumeric
isBool
isInt
isFloat

由于它将是一个解释型语言,出于效率考虑,我不会让它支持任何的语法糖,不支持多余的、景上添花的特性,避免在运行时过多影响效率。
得益于 Go 语言原生的一些高级特性,例如 管道、协程 这些可轻松实现异步操作的特性,Lit 会有更多扩展空间,它会比 php 更灵活、更小巧。

项目地址 github.com/pywee/lit