免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
1234下一页
最近访问板块 发新帖
查看: 13370 | 回复: 31
打印 上一主题 下一主题

[算法] 怎么解析一个数学表达式 [复制链接]

论坛徽章:
1
综合交流区版块每日发帖之星
日期:2015-10-14 06:20:00
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2013-10-18 11:23 |只看该作者 |倒序浏览
char s[64],  s里存了用户输入的一行字符,  这一行字符期望是数学运算表达式, 比如 1 + 2,   2 * 3 ,  4 * (5 + 7), 7 / (2 + 5)等等,  但用户也可能输入些无效的表达式.

怎么编程实现这一套表达式解析呢?  要求输入无效的时候给出错误提示,  其次要能解析括号,  + - * / 优先级,  最好浮点数也能认出来并加以运算?  

该用什么样的结构和算法呢?    看似简单, 但是一时居然也想不到什么方法.

论坛徽章:
324
射手座
日期:2013-08-23 12:04:38射手座
日期:2013-08-23 16:18:12未羊
日期:2013-08-30 14:33:15水瓶座
日期:2013-09-02 16:44:31摩羯座
日期:2013-09-25 09:33:52双子座
日期:2013-09-26 12:21:10金牛座
日期:2013-10-14 09:08:49申猴
日期:2013-10-16 13:09:43子鼠
日期:2013-10-17 23:23:19射手座
日期:2013-10-18 13:00:27金牛座
日期:2013-10-18 15:47:57午马
日期:2013-10-18 21:43:38
2 [报告]
发表于 2013-10-18 11:40 |只看该作者
主要用栈  

论坛徽章:
17
处女座
日期:2013-08-27 09:59:352015亚冠之柏太阳神
日期:2015-07-30 10:16:402015亚冠之萨济拖拉机
日期:2015-07-29 18:58:182015年亚洲杯之巴勒斯坦
日期:2015-03-06 17:38:17摩羯座
日期:2014-12-11 21:31:34戌狗
日期:2014-07-20 20:57:32子鼠
日期:2014-05-15 16:25:21亥猪
日期:2014-02-11 17:32:05丑牛
日期:2014-01-20 15:45:51丑牛
日期:2013-10-22 11:12:56双子座
日期:2013-10-18 16:28:17白羊座
日期:2013-10-18 10:50:45
3 [报告]
发表于 2013-10-18 11:46 |只看该作者
本帖最后由 myworkstation 于 2013-10-18 11:49 编辑

回复 1# fly3ds


    二楼说的对,你可以参考这个库的实现:http://www.gnu.org/software/libmatheval/
    你可以去看看数据结构与算法中的“中缀表达式”,“后缀表达式”

论坛徽章:
59
2015年亚洲杯之约旦
日期:2015-01-27 21:27:392015年亚洲杯之日本
日期:2015-02-06 22:09:41拜羊年徽章
日期:2015-03-03 16:15:432015年辞旧岁徽章
日期:2015-03-03 16:54:152015年迎新春徽章
日期:2015-03-04 09:50:282015元宵节徽章
日期:2015-03-06 15:50:392015年亚洲杯之阿联酋
日期:2015-03-19 17:39:302015年亚洲杯之中国
日期:2015-03-23 18:52:23巳蛇
日期:2014-12-14 22:44:03双子座
日期:2014-12-10 21:39:16处女座
日期:2014-12-02 08:03:17天蝎座
日期:2014-07-21 19:08:47
4 [报告]
发表于 2013-10-18 11:58 |只看该作者
LR (1). or LL(1)

论坛徽章:
12
巳蛇
日期:2013-09-16 15:32:242015年辞旧岁徽章
日期:2015-03-03 16:54:152015年亚洲杯之约旦
日期:2015-02-11 14:38:37双鱼座
日期:2015-01-05 11:05:47戌狗
日期:2014-12-08 09:41:18戌狗
日期:2014-08-15 09:29:29双子座
日期:2014-08-05 09:17:17卯兔
日期:2014-06-08 15:32:18巳蛇
日期:2014-01-27 08:47:08白羊座
日期:2013-11-28 21:04:15巨蟹座
日期:2013-11-13 21:58:012015年亚洲杯之科威特
日期:2015-04-17 16:51:51
5 [报告]
发表于 2013-10-18 12:05 |只看该作者
可以参考lua源代码中lparser.c中的subexpr函数的实现,这个函数用的方法在手写递归下降的编译器中挺常见,也不是很复杂。

论坛徽章:
0
6 [报告]
发表于 2013-10-18 12:49 |只看该作者
其实真的不简单

论坛徽章:
1
射手座
日期:2013-08-21 13:11:46
7 [报告]
发表于 2013-10-18 13:11 |只看该作者
LL
+1

lol

论坛徽章:
9
摩羯座
日期:2013-08-15 15:18:48狮子座
日期:2013-09-12 18:07:47金牛座
日期:2013-09-16 13:23:09辰龙
日期:2013-10-09 09:03:27白羊座
日期:2013-10-17 13:32:44子鼠
日期:2014-04-23 15:09:38戌狗
日期:2014-09-17 11:37:542015年亚洲杯之韩国
日期:2015-03-26 10:16:442015亚冠之武里南联
日期:2015-08-18 14:55:52
8 [报告]
发表于 2013-10-18 14:18 |只看该作者
本帖最后由 w_anthony 于 2013-10-18 14:20 编辑

我怎么记得是前缀表达式可以比较轻松的解决这个问题,粗略的想了一下也应该是前缀表达式。
先有操作符,再取操作数,如果操作数仍然是一个操作符,那么递归进去,应该是比较简单的。
问题是只要解析生成这样的一个前缀表达式就OK了,记得可以通过生成关于表达式的一颗树,然后通过前序遍历就可以得到前缀表达式了。

论坛徽章:
8
CU大牛徽章
日期:2013-04-17 10:59:39CU大牛徽章
日期:2013-04-17 11:01:45CU大牛徽章
日期:2013-04-17 11:02:15CU大牛徽章
日期:2013-04-17 11:02:36CU大牛徽章
日期:2013-04-17 11:02:58技术图书徽章
日期:2013-12-04 10:48:50酉鸡
日期:2014-01-03 10:32:30辰龙
日期:2014-03-06 15:04:07
9 [报告]
发表于 2013-10-18 15:03 |只看该作者
本帖最后由 shan_ghost 于 2013-10-18 15:05 编辑

偶前些天搞过一个……用来做动态调度的……

做法是通过简单解析找到其中的未知数,然后套个c的函数壳丢给gcc编译成动态链接库(太没节操了……);最后在载入模块做了一套简易版本管理逻辑去触发自动编译,以保证动态库总是最新的……

另外还搞了些乱七八糟的机制,以便在真正调用时识别参数个数和类型……

论坛徽章:
9
摩羯座
日期:2013-08-15 15:18:48狮子座
日期:2013-09-12 18:07:47金牛座
日期:2013-09-16 13:23:09辰龙
日期:2013-10-09 09:03:27白羊座
日期:2013-10-17 13:32:44子鼠
日期:2014-04-23 15:09:38戌狗
日期:2014-09-17 11:37:542015年亚洲杯之韩国
日期:2015-03-26 10:16:442015亚冠之武里南联
日期:2015-08-18 14:55:52
10 [报告]
发表于 2013-10-18 15:11 |只看该作者
回复 9# shan_ghost


    我只能说能想到这么搞的你,不管实际效果如何,都是非常非常的牛B!
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP