Skip to content

Commit f882a4a

Browse files
author
wanghy
committed
init: init
0 parents  commit f882a4a

16 files changed

+1237
-0
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
.idea

README.md

+122
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,122 @@
1+
## **规则**
2+
建议使用其他md查看器查看。
3+
4+
### **基础运算**
5+
6+
基础运算符表:
7+
8+
| 符号 | 意义 |
9+
| :--: | :------: |
10+
| + ||
11+
| - ||
12+
| * ||
13+
| / ||
14+
| ^ | 乘方 |
15+
| ( | 左括号 |
16+
| ) | 右括号 |
17+
| & ||
18+
| \| ||
19+
|||
20+
| = | 等于 |
21+
| > | 大于 |
22+
| >= | 大于等于 |
23+
| < | 小于 |
24+
| <= | 小于等于 |
25+
26+
27+
28+
### **变量**
29+
30+
变量使用`{}`包裹,如`{count}`表示`count`变量。变量由字母、数字、`_`组成。首写字符必须是`_`或者字母。
31+
32+
### **函数**
33+
34+
只有函数库中存在的函数才可以使用。函数不区分大小写。如:`MAX` `max` `Max` `mAx`都表示函数`MAX`
35+
36+
函数在构建ast树、运算时都会进行参数个数校验,保证运行安全。
37+
38+
#### **增加函数步骤**
39+
40+
1.`func.go`中添加函数方法,参数、返回体必须是固定的格式
41+
2.`vars.go/FuncMap`中添加函数名和对应方法
42+
3.`vars.go/FuncParNumMap`中添加函数的参数个数。
43+
44+
#### **函数参数特定值**
45+
46+
- `Abt` 任意个
47+
- `GtZero` 大于0个
48+
49+
## **BNF**
50+
51+
此处记录BNF式。
52+
53+
> BNF 规则:
54+
>
55+
> 在双引号中的字("word")代表着这些字符本身。而double_quote用来代表双引号。
56+
>
57+
> 在双引号外的字(有可能有下划线)代表着语法部分。
58+
>
59+
> 尖括号( < > )内包含的为必选项。
60+
>
61+
> 方括号( [ ] )内包含的为可选项。
62+
>
63+
> 大括号( { } )内包含的为可重复0至无数次的项。
64+
>
65+
> 竖线( | )表示在其左右两边任选一项,相当于"OR"的意思。
66+
>
67+
> ::= 是“被定义为”的意思。
68+
69+
优先级(低到高):`|,&,!,{>,<,=,>=,<=},{+,-},{*,/},^,func,()`
70+
71+
越底层优先级越高
72+
73+
```BNF
74+
<expr> ::= <and_term> { OR <expr> }
75+
<and_term> ::= <not_expr> { AND <and_expr>}
76+
<not_term> ::= { NOT } <com_expr>
77+
<com_term> ::= <pri_pre> { GT|LT|EQ|NEQ|GTE|LTE <pri_pre> } // compare term
78+
<pri_ope> ::= { <pri_ope> +|- } <sec_ope> // Primary operation
79+
<sec_ope> ::= { <sec_ope> *|/ } <ter_ope> // Secondary operation
80+
<ter_ope> ::= <factor> { ^ <ter_ope> } // Tertiary operation
81+
<factor> ::= NUM|
82+
FUNCTION LPAREN [ expr { COMMA expr }] RPAREN|
83+
IDENTIFIER|
84+
{ PLUS | MINUS } factor|
85+
LPAREN expr RPAREN
86+
```
87+
88+
## **使用**
89+
90+
`enter.go`中提供了`GetAstTreeByString()``CalByAstTree`方法。
91+
92+
- `GetAstTreeByString`:参数为字符串,返回ast根节点和err。
93+
- `CalByAstTree`:参数为ast树节点,返回`*decimal.Decimal`类型数据和err
94+
95+
样例:
96+
97+
```golang
98+
func TestDemo(t *testing.T) {
99+
str := "IF(1+5*9=8/8|2-1<5*1&1+2!=8*5,5+(7*7+2),9998848.24425)"
100+
node, err := calculator.GetAstTreeByString(str)
101+
if err != nil {
102+
t.Error(err)
103+
return
104+
}
105+
res, err := calculator.CalByAstTree(node, nil)
106+
println(res.String())
107+
println(res.Float64())
108+
println(res.IntPart())
109+
}
110+
```
111+
112+
print:
113+
114+
```bash
115+
=== RUN TestDemo
116+
56
117+
+5.600000e+001 true
118+
56
119+
--- PASS: TestDemo (0.00s)
120+
PASS
121+
```
122+

ast_node.go

+99
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,99 @@
1+
package formula_engine
2+
3+
type AstNode interface {
4+
GetName() string
5+
GetTok() *token
6+
}
7+
8+
// astGeneralNode 一般树节点
9+
type astGeneralNode struct {
10+
Tok *token
11+
Nodes []AstNode
12+
}
13+
14+
func newAstGeneralNode(t *token, node ...AstNode) *astGeneralNode {
15+
return &astGeneralNode{
16+
Tok: t,
17+
Nodes: node,
18+
}
19+
}
20+
21+
func (a *astGeneralNode) GetName() string {
22+
return "astGeneralNode"
23+
}
24+
25+
func (a *astGeneralNode) GetTok() *token {
26+
return a.Tok
27+
}
28+
29+
//--------------------------------------------------------------------------------
30+
//--------------------------------------------------------------------------------
31+
32+
// astBinNode BinaryNode 二叉树
33+
type astBinNode struct {
34+
Tok *token
35+
LNode AstNode
36+
RNode AstNode
37+
}
38+
39+
func newAstBinNode(t *token, lNode AstNode, rNode AstNode) *astBinNode {
40+
return &astBinNode{
41+
Tok: t,
42+
LNode: lNode,
43+
RNode: rNode,
44+
}
45+
}
46+
47+
func (a *astBinNode) GetName() string {
48+
return "astBinNode"
49+
}
50+
51+
func (a *astBinNode) GetTok() *token {
52+
return a.Tok
53+
}
54+
55+
//--------------------------------------------------------------------------------
56+
//--------------------------------------------------------------------------------
57+
58+
// astUnNode UnaryNode,单叉树
59+
type astUnNode struct {
60+
Tok *token
61+
Node AstNode
62+
}
63+
64+
func newAstUnNode(t *token, node AstNode) *astUnNode {
65+
return &astUnNode{
66+
Tok: t,
67+
Node: node,
68+
}
69+
}
70+
71+
func (a *astUnNode) GetName() string {
72+
return "astUnNode"
73+
}
74+
75+
func (a *astUnNode) GetTok() *token {
76+
return a.Tok
77+
}
78+
79+
//--------------------------------------------------------------------------------
80+
//--------------------------------------------------------------------------------
81+
82+
// astSinNode single,单个节点
83+
type astSinNode struct {
84+
Tok *token
85+
}
86+
87+
func newAstSinNode(t *token) *astSinNode {
88+
return &astSinNode{
89+
Tok: t,
90+
}
91+
}
92+
93+
func (a *astSinNode) GetName() string {
94+
return "astSinNode"
95+
}
96+
97+
func (a *astSinNode) GetTok() *token {
98+
return a.Tok
99+
}

0 commit comments

Comments
 (0)