前几天看一个网友提问,如何计算'1+1'这种字符串的值,不能用eval函数.

我仿佛记得以前新手时,对这个问题完全不知道如何下手.

我觉得处理括号实在是太复杂了,多层嵌套括号怎么解析呢?一些多余的括号呢?

而在没有括号的情况下,处理不同运算符之间的优先级又很头疼.

而'**'这种占2个字符的运算符,还有着奇特的优先级规则,x**y**z应先计算y**z,并且如果z后面还跟着**的话,则不能先计算y**z..这是最难的地方!

但是,今天我竟然把它实现了!而且是用最原始的函数实现的,它们是:

str,s.isdigit,s.find,s.count,
dict,float
if,in,and,not
==,>=,+(字符串连接)

float对象的6个方法不计算在内,这是无法避免的基础函数.

而str,float,dict只是起到类型转换或初始化数据的作用.所以真正用到的函数更是少之又少.

可能正是因为如此,我在程序设计上花了很多的心思.我用了非常多的if判断.显得有些难以看懂.而实际上每个分支都对应一种简单的情况.

唯一难的地方在于幂的递归运算(get_rig_rest函数).

这令我非常激动.这让我意识到自己解决复杂问题的能力!

另外应该指出,我的思路来自于前段时间学习Scheme时重写一些基本函数所养成的递归思考的习惯.

同时,Python真的是太方便和强大了,快速的print让测试+修改十分简单,极大提高了大脑解决抽象问题的能力(没有print,也许你要浪费很多笔,纸和时间).

dic={'+':float.__add__,
'-':float.__sub__,
'*':float.__mul__,
'/':float.__floordiv__,
'%':float.__mod__,
'**':float.__pow__,} priority=dict((('**',3),('*',2),('/',2),('%',2),('+',1),('-',1))) def prt(opf,p_opf):
'幂运算符总是最优先,即使左边也是幂运算符'
if p_opf=='**':
return False
return priority[opf]>=priority[p_opf] def get_val(lef,opf,rig):
return str(dic[opf](float(lef),float(rig))) def isfloat(s):
return s.isdigit() or s=='.' def get_brace(s,start=0):
pos=s.find(')',start)+1
brace=s[:pos]
if brace.count('(')==brace.count(')'):
return brace,s[pos:]
return get_brace(s,pos) def get_rig_rest(s,num=''):
'分裂为数字+剩余部分'
'如果数字后是幂运算符,则需要计算该数字的幂,将结果作为数字部分返回'
if s[:2] =='**':
pow_num,rest=get_rig_rest(s[2:])
return get_val(num,'**',pow_num),rest
if s=='' or s[0] in dic:
return num,s
if s[0]=='(':
brace,rest=get_brace(s)
return ieval(brace[1:-1]),rest
if isfloat(s[0]):
return get_rig_rest(s[1:],num+s[0]) def ieval(s,lef='',opf=None,rig=''):
if not opf and not s:
return lef
if not opf and isfloat(s[0]):
return ieval(s[1:],lef+s[0],opf,rig)
if not opf and s[0] in dic:
if s[1]=='*':
return ieval(s[2:],lef,'**',rig)
return ieval(s[1:],lef,s[0],rig)
if not opf and s[0]=='(':
brace,rest=get_brace(s)
if not rest:
return ieval(brace[1:-1])
return ieval(rest,lef+ieval(brace[1:-1]),
opf,rig)
if opf and not s:
return get_val(lef,opf,rig)
if opf and s[0]=='(':
brace,rest=get_brace(s)
if not rest:
return get_val(lef,opf,ieval(brace[1:-1]))
return ieval(lef+opf+ieval(brace[1:-1])+rest,
'',None,'')
if opf and isfloat(s[0]):
return ieval(s[1:],lef,opf,rig+s[0])
if opf and s[0] in dic:
number=2 if s[1]=='*' else 1
p_opf=s[:number]
if prt(opf,p_opf):
return ieval(s,get_val(lef,opf,rig),None,'')
p_rig,p_rest=get_rig_rest(s[number:])
return ieval(lef+opf+get_val(rig,p_opf,p_rig)+p_rest,
'',None,'') test=['1+2+3', #常规
'1+2*9', #简单运算符优先级
'1+2*(3+4)', #简单运算符优先级+括号
'4**3**2', #幂运算优先级
'3+2*0.5**(2+2**2)',#多重优先级+括号
'(1+2)*(2+3)', #并列括号
'((9+3)/2)', #嵌套括号
'((1234)-1)', #多余括号
'((1+3**2)*((2%3+(3+3*2**3))*(4-1)*(1+2)+5*6)*4)'] #复杂情形 for x in test:
print(ieval(x),'---',eval(x))

结果:

>>>
6.0 --- 6
19.0 --- 19
15.0 --- 15
262144.0 --- 262144
3.03125 --- 3.03125
15.0 --- 15
6.0 --- 6.0
1233.0 --- 1233
11640.0 --- 11640

用Python最原始的函数模拟eval函数的浮点数运算功能的更多相关文章

  1. 用Python最原始的函数模拟eval函数的浮点数运算功能(2)

    这应该是我编程以来完成的难度最大的一个函数了.因为可能存在的情况非常多,需要设计合理的参数来控制解析流程.经验概要: 1.大胆假设一些子功能能够实现,看能否建立整个框架.如果在假设的基础上都无法建立, ...

  2. python中的exec()函数和eval()函数

    exec()函数 exec函数用于执行存储在字符串中的python语句 >>> exec("x=1") >>> x 但有时候,直接这样执行可能会 ...

  3. 编写函数模拟strcpy()函数功能

    strcpy(字符数组1,字符串2) strcpy( )用于将字符串2复制到字符数组1中 /* strcpy(字符数组1,字符串2) strcpy( )用于将字符串2复制到字符数组1中 模拟strcp ...

  4. R语言parse函数与eval函数的字符串转命令行及执行操作

    parse()函数能将字符串转换为表达式expression:eval()函数能对表达式求解 A <- : B <- 'print(A)' class(B) C <- parse(t ...

  5. Python使用map,reduce高阶函数模拟实现Spark的reduceByKey算子功能

    # 使用默认的高阶函数map和reduce import randomdef map_function(arg):  # 生成测试数据 return (arg,1) list_map = list(m ...

  6. python 内置函数input/eval(22)

    python的内置函数其实挺多的,其中input和eval算得上比较特殊,input属于交互式内置函数,eval函数能直接执行字符串表达式并返回表达式的值. 一.input函数 input是Pytho ...

  7. [Effective JavaScript 笔记]第17条:间接调用eval函数优于直接调用

    eval函数不仅仅是一个函数.大多数函数只访问定义它们所在的作用域,而不能访问除此之外的作用域(词法作用域).eval函数具有访问调用它时的整个作用域的能力.编译器编写者首次设法优化js时,eval函 ...

  8. Matlab匿名函数,子函数,私有函数,重载函数,eval和feval函数

    匿名函数,子函数,私有函数等函数类型 匿名函数: 匿名函数没有函数名,也不是.m文件,只包含一个表达式和输入输出参数. Fxy=@(x,y)x.^y+3*x*y x,y为输入输入参数,Fxy为函数名 ...

  9. python笔记-调用eval函数出现invalid syntax错误

    本来是想打算使用eval函数对变量进行赋值的,没想到出现了invalid syntax错误.源代码如下 In [2]: eval('a = 1') File "<string>& ...

随机推荐

  1. WebRTC 音频采样算法 附完整C++示例代码

    之前有大概介绍了音频采样相关的思路,详情见<简洁明了的插值音频重采样算法例子 (附完整C代码)>. 音频方面的开源项目很多很多. 最知名的莫过于谷歌开源的WebRTC, 其中的音频模块就包 ...

  2. 用Python浅析股票数据

    用Python浅析股票数据 本文将使用Python来可视化股票数据,比如绘制K线图,并且探究各项指标的含义和关系,最后使用移动平均线方法初探投资策略. 数据导入 这里将股票数据存储在stockData ...

  3. STM32 - GPIO

    买了一个STM32F4的开发板,想把上面的东西重新学一下 快速过: 一.GPIO控制 void GPIO_DeInit(GPIO_TypeDef* GPIOx); //把某一个IO口恢复到默认值 /* ...

  4. [HAOI2006]数字序列

    题目描述 现在我们有一个长度为n的整数序列A.但是它太不好看了,于是我们希望把它变成一个单调严格上升的序列.但是不希望改变过多的数,也不希望改变的幅度太大. 输入输出格式 输入格式: 第一行包含一个数 ...

  5. [HNOI2012]射箭

    Description 沫沫最近在玩一个二维的射箭游戏,如下图 1 所示,这个游戏中的 x 轴在地面,第一象限中有一些竖直线段作为靶子,任意两个靶子都没有公共部分,也不会接触坐标轴.沫沫控制一个位于( ...

  6. 计蒜客NOIP模拟赛D2T2 直线的交点

    伦伦刚刚在高中学习了解析几何,学会了计算两条直线的交点.这天,老师给她布置了一道作业.在平面上有 nnn 条直线,他们之间有若干交点.给定一对平板(两条平行的直线),问这有多少对直线,他们的交点在这一 ...

  7. 洛谷4月月赛R1

    T1.题目大意:n个人站成一排,有m个团队,每个人有且属于一个团队,可以让若干个人出队,任意交换这些人的位置后再站回去,问要让所有同一团队的人连续地站在一起,至少要出队几个.(n<=10^5,m ...

  8. hdu 5465 (树状数组 + 博弈)

    题意:基于矩阵的NIM游戏,求异或和. 思路:在x1,y1 到 x2, y2的异或和 =  A[ x2 ][ y2 ] ^ A[x1-1][ y2 ] ^ A[ x2 ][y1 - 1] ^ A[ x ...

  9. [BZOJ]3671 随机数生成器(Noi2014)

    洛谷上卡不过去的朋友们可以来看看小C的程序(小C才不是标题党呢!) Description Input 第1行包含5个整数,依次为 x_0,a,b,c,d ,描述小H采用的随机数生成算法所需的随机种子 ...

  10. 勤拂拭软件 java web 开发教程(1) - 开发环境搭建

    勤拂拭软件系列教程 之 Java Web开发之旅(1) Java Web开发环境搭建 1 前言 工作过程中,遇到不少朋友想要学习jsp开发,然而第一步都迈不出,连一个基本的环境都没有,试问,如何能够继 ...