解析数学表达式 代码解析AST语法树
2019年2月20日09:18:22
AST语法树自己写代码解析的话就比较麻烦,有现成的库可以解析PHP,就像webpack就是自己解析js的语法代码,编译成各种版本的可用代码
github https://github.com/josdejong/mathjs
| Extension | Description |
|---|---|
| mathsteps | A step-by-step math solver library that is focused on pedagogy (how best to teach). The math problems it focuses on are pre-algebra and algebra problems involving simplifying expressions. |
| mathjs‑expression‑parser | This custom build of mathjs contains just the expression parser and basic arithmetic functions for numbers. About four times as small as the full mathjs library. |
| mathjs-simple-integral | Extends Math.js to be able to compute simple integrals. |
| math.diff.js | Symbolic differentiation plugin for Math.js |
| postcss-math | PostCSS plugin for making calculations with math.js |
没有办法,自己去实现 前缀,中缀,后缀表达式来实现解析字符串,对于简单的加减乘除都是比较容易的,但是需要支持一些复杂一点逻辑的计算就比较麻烦,比如开方,乘方等
其他一些解析工具基本都是java ,c,cpp的
又尝试找了一些工具,发现JavaScript里面有一些,但是不符合我的个人需求,但是可以满足大部分,简单数学字符数解析和计算
PHP可用的库
composer require nikic/php-parser
一直在更新可以使用
namespace App\Http\Controllers\Data\V2; use App\Http\Controllers\Data\V2\BaseController as Base; use PhpParser\Error;
use PhpParser\NodeDumper;
use PhpParser\ParserFactory; class CommonController extends Base { public static function index(Request $Request) {
$code = <<<CODE
<?php
((99 + 1)*4-1);
CODE;
$parser = (new ParserFactory)->create(ParserFactory::PREFER_PHP7);
try {
$ast = $parser->parse($code);
} catch (Error $error) {
echo "Parse error: {$error->getMessage()}\n";
return;
} $dumper = new NodeDumper;
echo $dumper->dump($ast);
结果
我是使用pre打印的
array(
0: Stmt_Expression(
expr: Expr_BinaryOp_Minus(
left: Expr_BinaryOp_Mul(
left: Expr_BinaryOp_Plus(
left: Scalar_LNumber(
value: 99
)
right: Scalar_LNumber(
value: 1
)
)
right: Scalar_LNumber(
value: 4
)
)
right: Expr_BinaryOp_Plus(
left: Scalar_LNumber(
value: 1
)
right: Scalar_LNumber(
value: 2
)
)
)
)
)
基本就可以达到语法树解析,元素自己解析就可以了,但是支持的运算符只能支持官方已有的运算符,特殊的运算符或者自定义的运算符,得自己标记去解析,特别复杂的需要自己根据解析的ast去重新转换成自己实际业务的需求
如果你需要添加一个vistor 遍历真个ast树 ,并获取,修改数据
<?php namespace App\Service; use PhpParser\NodeVisitorAbstract;
use PhpParser\Node; class Visitor extends NodeVisitorAbstract { public $data;
public $operator; public function __construct() {
$this->data = new \SplStack();
$this->operator = new \SplStack();
} public function leaveNode(Node $node) {
//所有的符号
if ($node instanceof Node\Expr) {
$this->operator->push($node->getType());
}
//所有运算符
// if ($node instanceof Node\Expr\BinaryOp) {
// $this->operator->push($node->getType());
// }
// $this->operator->push($node->getType());
if ($node instanceof Node\Scalar) {
$this->data->push((string) $node->value);
}
} public function getData() {
return $this->data;
} public function getOperator() {
return $this->operator;
} }
然后在解析ast树的代码添加vistor
public function TransformToAst($string = '') {
try {
if (empty($string)) {
throw new \Exception('公式不能为空');
}
$code = <<<CODE
<?php
$string;
CODE;
$ParserFactory = new ParserFactory();
$parser = $ParserFactory->create(ParserFactory::PREFER_PHP7);
$ast = $parser->parse($code);
$traverser = new NodeTraverser;
$Visitor = new Visitor();
$traverser->addVisitor($Visitor);
$modifiedStmts = $traverser->traverse($ast);
p($Visitor->getOperator());
pp($Visitor->getData());
// pp($modifiedStmts);
//
die;
if (empty($ast_object)) {
throw new \Exception('解析表达式为空');
}
$ast_new = self::ParseAstToArray($ast_object['0']);
return $ast_new;
} catch (\Exception $e) {
throw new \Exception($e->getMessage() . $e->getFile() . $e->getLine());
}
}
其他提供的很多工具类,但是缺少demo实例,让处入手的人很难直接上手使用
参考 https://github.com/nikic/PHP-Parser/blob/master/doc/component/Walking_the_AST.markdown 以后有时间在翻译一下文档
Microsoft/tolerant-php-parser
这个微软的库
解析数学表达式 代码解析AST语法树的更多相关文章
- Vue源码解析:AST语法树转render函数
开始 今天要说的代码全在codegen文件夹中,在说实现原理前,还是先看个简单的例子! <div class="container"> <span>{{ms ...
- 机器学习完整过程案例分布解析,python代码解析
所谓学习问题,是指观察由n个样本组成的集合,并依据这些数据来预測未知数据的性质. 学习任务(一个二分类问题): 区分一个普通的互联网检索Query是否具有某个垂直领域的意图.如果如今有一个O2O领域的 ...
- C++实现对数学基本运算表达式的解析
代码地址如下:http://www.demodashi.com/demo/11078.html 前段时间在LeetCode上刷题,遇到了很多涉及对字符串进行解析的题目.可能是出于这个原因,最近迷恋上了 ...
- Babel(抽象语法树,又称AST)
文章:https://juejin.im/post/5a9315e46fb9a0633a711f25 https://github.com/jamiebuilds/babel-handbook/blo ...
- 笔记-爬虫-js代码解析
笔记-爬虫-js代码解析 1. js代码解析 1.1. 前言 在爬取网站时经常会有js生成关键信息,而且js代码是混淆过的. 以瓜子二手车为例,直接请求https://www.guaz ...
- 基于语法分析器GOLD Parser开发的数学表达式计算器
最近发现一款文法分析神器,看完官网(http://goldparser.org/)的介绍后感觉很犀利的样子,于是就拿来测试了一番,写了一个数学表达式分析的小程序,支持的数学运算符如下所示:常规运算:+ ...
- 抽象语法树 Abstract syntax tree
什么是抽象语法树? 在计算机科学中,抽象语法和抽象语法树其实是源代码的抽象语法结构的树状表现形式 在线编辑器 我们常用的浏览器就是通过将js代码转化为抽象语法树来进行下一步的分析等其他操作.所以将js ...
- WHY数学表达式的3D可视化
WHY数学表达式的3D可视化 很早之前我就有这种想法,将数学表达式的图形显示出来.最近终于实现了这套较为完善的版本,将其代码公布,也为开源做点贡献.首先系统中定义一套脚本语言格式,用于描述数学表达式. ...
- JavaScript的工作原理:解析、抽象语法树(AST)+ 提升编译速度5个技巧
这是专门探索 JavaScript 及其所构建的组件的系列文章的第 14 篇. 如果你错过了前面的章节,可以在这里找到它们: JavaScript 是如何工作的:引擎,运行时和调用堆栈的概述! Jav ...
随机推荐
- Celery - 一个懂得 异步任务 , 定时任务 , 周期任务 的芹菜
1.什么是Celery?Celery 是芹菜Celery 是基于Python实现的模块, 用于执行异步定时周期任务的其结构的组成是由 1.用户任务 app 2.管道 broker 用于存储 ...
- ionic 扫描二维码 Barcode Scanner、QR Scanner、ZBar
1.简介 ionic 官方给我们提供了三个扫描二维码条形码插件,分别为: Barcode Scanner 样式好看,类似支付宝的扫描框.速度稍微比最后一个慢几百毫秒,主要问题是 Android 部分手 ...
- 🍓 移动端调试工具之vconsole的使用~ 🍓
这里以在vue项目中的使用为例⬇️ 嗯模块化的. 不消多说,先cnpm install vconsole -S 然后在mian.js中配置之- ok啦-- 开发混合app的筒子,使用mac的话也有别的 ...
- php in_array() 循环大量数组时效率特别慢问题
in_array() 会循环数组内部元素逐个匹配,特别耗时,换成以下方式,效率大大提升
- java web中使用mysql语句遇到的问题
1.插入数据时遇到 Parameter index out of range (1 > number of parameters, which is 0). 的问题 有问题的代码: 改 ...
- having的用法
转载:http://blog.csdn.net/oathevil/article/details/5521757 where和having: “Where” 是一个约束声明,使用Where来约束来自于 ...
- Hibernate查询返回自定义VO的两种方式
说明:createQuery用的hql语句进行查询,createSQLQuery用sql语句查询: 前者以hibernate生成的Bean为对象装入list返回:后者则是以对象数组进行存储: 一.通过 ...
- .Net Core---- 自带Json返回日期带T格式 解决
前段时间再做core的列表显示中(前台代码是在.net core bootstrap集成框架上的(这是效果浏览地址:http://core.jucheap.com[效果地址来自:http://blog ...
- 洛谷P5206 [WC2019]数树 [容斥,DP,生成函数,NTT]
传送门 Orz神仙题,让我长了许多见识. 长式子警告 思路 y=1 由于y=1时会导致后面一些式子未定义,先抓出来. printf("%lld",opt==0?1:(opt==1? ...
- poj 1426 Find The Multiple (简单搜索dfs)
题目: Given a positive integer n, write a program to find out a nonzero multiple m of n whose decimal ...