`
drift_ice
  • 浏览: 9598 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
最近访客 更多访客>>
文章分类
社区版块
存档分类
最新评论

表达式解析器

阅读更多

 今天逛javaEye看到久违的逆波兰式又重出江湖(见 http://www.iteye.com/topic/717341),恰好毕业一年快一个月了,决定写一篇技术BLO纪念一下渐行渐远的大学时光。


这个解析器支持以下几个功能:

1.负数

2.四则运算

 

3.支持变量

3.自定义函数


原理是逆波兰式,结构是设计模式里的解释器模式,调用代码如下:

 

public static void main(String[] args) {
		String exp = "mod(11,3)+2*a*t*t";
		try {
			Node expNode = parse(exp);
			expNode.setVariable("a", 10);
			expNode.setVariable("t", 2);
			if (expNode != null)
				System.out.println(expNode + "=" + expNode.getValue());
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

 

 

一.类结构如下:


类结构

 

Node主要有三个方法

interface Node {
 /**
  * 返回表达式
  * @return
  */
 public String toString();

 /**
  * 设置变量值
  * @param var
  * @param value
  */
 public void setVariable(String var, double value);

 /**
  * 返回值
  * @return
  */
 public double getValue();
}

 

OptNode和NumNode分别标识操作符节点和数值节点

OptNode会多一个public int getPriority()方法返回优先级。

 

二.主要的函数有两个

1.解析节点:

    这个方法的功能是解析字符串,根据字符串的内容生成相应的结点

public static Node parse(String exp) throws Exception {// 注册函数
		registerFuns();
		char[] ch = exp.toCharArray();
		Stack<Node> nodes = new Stack<Node>();
		for (int i = 0; i < ch.length;) {
			char c = ch[i];
			if (isSpace(c)) {
				i += 1;
				continue;
			} else if (isNum(nodes, c)) {
				String num = readNum(ch, i);
				nodes.push(value(Double.valueOf(num)));
				i += num.length();
			} else if (isOpt(c)) {
				Node opt = null;
				switch (c) {
				case '+':
					opt = new AndOptNode();
					break;
				case '-':
					opt = new SubOptNode();
					break;
				case '*':
					opt = new MultiOptNode();
					break;
				case '/':
					opt = new DivOptNode();
					break;
				case ',':
					opt = new CommaOptNode();
					break;
				default:
					break;
				}
				if (opt != null) {
					nodes.push(opt);
					i += 1;
				}
			} else if (c == ')') {
				List<Node> nodesInBracket = new ArrayList<Node>();
				Node node = nodes.pop();
				while (!nodes.isEmpty() && !(node instanceof LeftBracket)) {
					nodesInBracket.add(0, node);
					node = nodes.pop();
				}

				BracketNode result = compute(nodesInBracket);
				/*
				 * 如果前一个点是函数,则将其设为函数的参数,否则,则作为一般的节点
				 */
				if (!nodes.isEmpty() && nodes.peek() instanceof FunctionNode) {
					FunctionNode fun = (FunctionNode) nodes.peek();
					fun.setParamNode(result);
				} else {
					nodes.push(result);
				}
				i += 1;
			} else if (c == '(') {
				nodes.push(leftBracket());
				i += 1;
			} else if (isVarStartChar(c)) {
				String indentifier = readIndentifier(ch, i);
				if (isFun(indentifier)) {
					FunctionNode fun = newFun(indentifier);
					nodes.push(fun);
				} else {
					nodes.push(var(indentifier));
				}
				i += indentifier.length();
			}
		}
		List<Node> buf = new ArrayList<Node>();
		while (!nodes.isEmpty()) {
			buf.add(0, nodes.pop());
		}
		if (buf.size() > 0) {
			BracketNode result = compute(buf);
			return result.getNode();
		} else
			return null;
	}

 

2.计算结点

这个方法的作用就是计算一系列不带括号的节点

 

private static BracketNode compute(List<Node> nodes) throws Exception {
		Stack<Node> stack = new Stack<Node>();
		Stack<OptNode> opts = new Stack<OptNode>();
		for (int i = 0; i < nodes.size(); i++) {
			Node node = nodes.get(i);
			if (node instanceof OptNode) {
				OptNode optNode = (OptNode) node;
				if (!opts.isEmpty()) {
					OptNode preNode = opts.peek();
					if (preNode.getPriority() >= optNode.getPriority()) {
						opts.pop();
						merge(stack, preNode);
					}
				}
				opts.push(optNode);
			} else {
				stack.push(node);
			}
		}
		while (!opts.isEmpty()) {
			merge(stack, opts.pop());
		}
		if (stack.size() != 1) {
			throw new Exception("express error");
		}
		return bracket(stack.pop());
	}

 
 3.其他

 

自定函数就是匹配标识符与注册的函数名,如果有,则利用反射生成相应的函数结点,如果没有,则作为普通的变量标识符使用。

可以像这样注册函数

private static void registerFuns() {
		registerFun("mod", ModFunction.class);
		registerFun("sqrt", SqrtFunction.class);
	}

 

 

第一次发这种内容复杂的贴,排版有点乱,注释有点少,将就着看吧。

 

图方便,把类都丢在一个类文件里,具体的代码见附近

 

  • 大小: 99.8 KB
分享到:
评论

相关推荐

    表达式表达式解析器表达式解析器

    表达式表达式解析器表达式解析器表达式解析器表达式解析器表达式解析器

    delphi 教你如何做表达式解析器

    在IT领域,构建一个表达式解析器是一项基础且重要的任务,尤其对于编程语言的设计与实现。本教程将通过Delphi这一强大的Object Pascal开发环境,教你如何构建一个表达式解析器,涉及词法分析器和语法分析器的关键...

    C# 表达式解析器源代码(修正)

    在本文中,我们将深入探讨C#中的表达式解析器,特别是在修正了算符优先级错误之后的实现。C#表达式解析器是一个程序,它能够分析输入的字符串表达式,并将其转换为计算机可以理解的形式,以便执行计算或逻辑操作。 ...

    表达式解析器源代码(可以计算各种函数).zip

    《深入解析表达式解析器:计算各种函数的源代码实现》 在计算机科学领域,表达式解析器是一种至关重要的工具,它能够理解并处理各种数学和逻辑表达式,执行计算并返回结果。这个名为"表达式解析器源代码(可以计算...

    IK Expression开源表达式解析器 V2.1.2.rar

    IK Expression开源表达式解析器V2.1.2是一款用于处理和解析文本表达式的强大工具,广泛应用于各类软件开发和数据分析场景。它以其高效、灵活和可扩展性著称,为开发者提供了便捷的方式来处理和执行复杂的逻辑表达式...

    四则运算表达式解析器源码(C#)

    标题中的“四则运算表达式解析器源码(C#)”是指一个使用C#编程语言编写的程序,它的主要功能是解析包含加法(+)、减法(-)、乘法(*)、除法(/)、取模(%)以及比较运算符(&lt;、&gt;、=、、、&gt;=)的数学表达式。...

    山寨版的简易表达式解析器

    标题 "山寨版的简易表达式解析器" 暗示我们即将探讨的是一个自定义的、简化版的表达式解析器实现。这类工具通常用于处理数学或逻辑表达式,将字符串形式的表达式转化为可执行的代码。在这个案例中,可能是为了学习...

    C#表达式解析器-1.0

    《C#表达式解析器-1.0:深入解析与应用》 在编程领域,表达式解析器是一种至关重要的工具,它能够理解并处理由程序员编写的数学或逻辑表达式,将其转化为计算机可执行的形式。C#表达式解析器正是这样一种实现,尤其...

    开源表达式解析器开源表达式解析器开源表达式解析器

    ### 开源表达式解析器知识点总结 #### 一、IKExpression概述 IKExpression是一个开源的、可扩展的、基于Java语言开发的超轻量级公式化语言解析执行工具包。该工具包完全独立,不依赖任何第三方Java库,能够轻松...

    java编写的正则表达式解析器

    在Java编程语言中,实现正则表达式解析器可以帮助我们更高效地处理字符串,理解正则表达式的内部工作原理,并且可以自定义扩展其功能。本文将深入探讨Java中正则表达式的相关知识。 首先,Java中的正则表达式主要...

    C# 表达式解析器源代码

    在IT领域,编程语言的表达式解析器是一个关键组件,它负责将人类可读的代码转化为计算机可执行的指令。本项目"**C# 表达式解析器源代码**"提供了一个用C#实现的简单表达式解析器,特别强调了对算术运算符的优先级...

    数学表达式解析器java语言描述

    数学表达式解析器是计算机科学中的一个重要工具,它允许程序理解和计算由人类编写的数学表达式。在Java语言中,实现这样一个解析器涉及到多个关键概念和技术。本篇将深入探讨这些知识点。 首先,数学表达式解析器的...

    多赋值语句算术表达式解析器2.2.rar_表达式解析_表达式解析 C_表达式计算器_解析表达式

    本项目"多赋值语句算术表达式解析器2.2"就是一个专注于处理这种任务的工具,尤其在C语言环境下。这个解析器能够处理包含多个赋值语句以及复杂算术和函数表达式的计算。 首先,让我们了解一下"表达式解析"。在编程中...

    简单实用的表达式解析器

    在IT领域,表达式解析器是一种至关重要的工具,它能够理解和转换输入的数学或逻辑表达式,以便计算机能够执行这些表达式。在这个“简单实用的表达式解析器”项目中,我们可以从给定的文件中了解到一些核心概念和技术...

    Java编程艺术-表达式解析器.rar

    本话题聚焦于Java编程艺术中的一个关键环节——表达式解析器,它允许我们处理和求值数学或逻辑表达式。在给定的“Java编程艺术-表达式解析器.rar”压缩包文件中,我们可以深入学习如何在Java中构建这样一个工具,以...

    delphi正则表达式解析器

    Delphi正则表达式解析器是一款专为Delphi开发者设计的工具,用于处理和解析正则表达式。这款解析器提供了源代码,使得开发者能够深入理解其内部工作原理,并可以根据需求进行定制或扩展。在Delphi编程环境中,正则...

    Applet表达式解析器.rar_表达式解析

    这个名为"Applet表达式解析器"的项目显然专注于处理数学表达式的解析,特别是使用了递归下降解析法来实现。递归下降解析是编译原理中的一个关键概念,常用于解释性或编译性语言的语法分析阶段,它将语法规则转化为一...

    CronExpression一个cron表达式解析器

    1. **解析cron表达式**:将字符串形式的cron表达式转化为可操作的对象,这样就可以方便地获取下一次执行时间或者判断当前时间是否满足表达式。 2. **验证cron表达式**:检查输入的表达式是否有效,避免因为错误的...

Global site tag (gtag.js) - Google Analytics