[Python标准库]decimal——定点数和浮点数的数学运算
作用:使用定点数和浮点数的小数运算。
Python 版本:2.4 及以后版本
decimal 模块实现了定点和浮点算术运算符,使用的是大多数人所熟悉的模型,而不是程序员熟悉的模型,即大多数计算机硬件实现的 IEEE 浮点数运算。Decimal 实例可以准确地表示任何数,对其上取整或下取整,还可以对有效数字个数加以限制。
Decimal
小数值表示为 Decimal 类的实例。构造函数取一个整数或字符串作为参数。使用浮点数创建 Decimal 之前,可以先将浮点数转换为一个字符串,使调用者能够显式地处理值得位数,倘若使用硬件浮点数表示则无法准确地表述。另外,利用类方法 from_float()
可以转换为精确的小数表示。
import decimal fmt = '{0:<25} {1:<25}' print fmt.format('Input', 'Output') print fmt.format('-' * 25, '-' * 25) # Integer print fmt.format(5, decimal.Decimal(5)) # String print fmt.format('3.14', decimal.Decimal('3.14')) # Float f = 0.1 print fmt.format(repr(f), decimal.Decimal(str(f))) print fmt.format('%.23g' % f, str(decimal.Decimal.from_float(f))[:25])
浮点数值 0.1 并不表示为一个精确的二进制值,所以 float 的表示与 Decimal 值不同。在这个输出中它被截断为 25 个字符。
Decimal 还可以由元组创建,其中包含一个符号标志(0 表示正,1 表示负)、数字 tuple 以及一个整数指数。
import decimal # Tuple t = (1, (1, 1), -2) print 'Input :', t print 'Decimal:', decimal.Decimal(t)
基于元组的表示创建时不太方便,不过它提供了一种可移植的方式,可以导出小数值而不会损失精度。tuple 形式可以在网络上传输,或者在不支持精确小数值得数据库中存储,以后再转回回 Decimal 实例。
算术运算
Decimal 重载了简单的算术运算符,所以可以采用内置数值类型同样的方式处理 Decimal 实例。
import decimal a = decimal.Decimal('5.1') b = decimal.Decimal('3.14') c = 4 d = 3.14 print 'a =', repr(a) print 'b =', repr(b) print 'c =', repr(c) print 'd =', repr(d) print print 'a + b =', a + b print 'a - b =', a - b print 'a * b =', a * b print 'a / b =', a / b print print 'a + c =', a + c print 'a - c =', a - c print 'a * c =', a * c print 'a / c =', a / c print print 'a + d =', try: print a + d except TypeError, e: print e
Decimal 运算符还接受整数参数,不过浮点数值必须转换为 Decimal 实例。
除了基本算术运算,Decimal 还包括一些方法来查找以 10 为底的对数和自然对数。log10() 和 ln() 返回的值都是 Decimal 实例,所以可以与其他值一样直接在公式中使用。
特殊值
除了期望的数字值,Decimal 还可以表示很多特殊值,包括正负无穷大值、“不是一个数”(NaN)和 0。
import decimal for value in [ 'Infinity', 'NaN', '0' ]: print decimal.Decimal(value), decimal.Decimal('-' + value) print # Math with infinity print 'Infinity + 1:', (decimal.Decimal('Infinity') + 1) print '-Infinity + 1:', (decimal.Decimal('-Infinity') + 1) # Print comparing NaN print decimal.Decimal('NaN') == decimal.Decimal('Infinity') print decimal.Decimal('NaN') != decimal.Decimal(1)
与无穷大值相加会返回另一个无穷大值。与 NaN 比较相等性总会返回 false,而比较不等性总会返回 true。与 NaN 比较大小来确定排序顺序没有明确定义,这会导致一个错误。
上下文
到目前为止,前面的例子使用的都是 decimal 模块的默认行为。还可以使用一个上下文(context)覆盖某些设置,如保持精度、如何完成取整、错误处理等等。上下文可以应用于一个线程中的所有 Decimal 实例,或者局部应用于一个小代码区。
1. 当前上下文
要获取当前全局上下文,可以使用 getcontext()。
import decimal import pprint context = decimal.getcontext() print 'Emax =', context.Emax print 'Emin =', context.Emin print 'capitals =', context.capitals print 'prec =', context.prec print 'rounding =', context.rounding print 'flags =' pprint.pprint(context.flags) print 'traps =' pprint.pprint(context.traps)
这个示例脚本显示了 Context 的公共属性。
2. 精度
上下文的 prec 属性控制着作为算术运算结果所创建的新值的精度。字面量值会按这个属性保持精度。
import decimal d = decimal.Decimal('0.123456') for i in range(4): decimal.getcontext().prec = i print i, ':', d, d * 1
要改变精度,可以直接为这个属性赋一个新值。
3. 取整
取整有多种选择,以保证值在所需精度范围内。
"htmlcode">
import decimal context = decimal.getcontext() ROUNDING_MODES = [ 'ROUND_CEILING', 'ROUND_DOWN', 'ROUND_FLOOR', 'ROUND_HALF_DOWN', 'ROUND_HALF_EVEN', 'ROUND_HALF_UP', 'ROUND_UP', 'ROUND_05UP', ] header_fmt = '{:10} ' + ' '.join(['{:^8}'] * 6) print header_fmt.format(' ', '1/8 (1)', '-1/8 (1)', '1/8 (2)', '-1/8 (2)', '1/8 (3)', '-1/8 (3)', ) for rounding_mode in ROUNDING_MODES: print '{0:10}'.format(rounding_mode.partition('_')[-1]), for precision in [ 1, 2, 3 ]: context.prec = precision context.rounding = getattr(decimal, rounding_mode) value = decimal.Decimal(1) / decimal.Decimal(8) print '{0:^8}'.format(value), value = decimal.Decimal(-1) / decimal.Decimal(8) print '{0:^8}'.format(value), print
这个程序显示了使用不同算法将同一个值取整为不同精度的效果。
4. 局部上下文
使用 Python 2.5 或以后版本时,可以使用 with 语句对一个代码块应用上下文。
import decimal with decimal.localcontext() as c: c.prec = 2 print 'Local precision:', c.prec print '3.14 / 3 =', (decimal.Decimal('3.14') / 3) print print 'Default precision:', decimal.getcontext().prec print '3.14 / 3 =', (decimal.Decimal('3.14') / 3)
Context 支持 with 使用的上下文管理器 API,所以这个设置只在块内应用。
5. 各实例上下文
上下文还可以用来构造 Decimal 实例,然后可以从这个上下文继承精度和转换的取整参数。
import decimal # Set up a context with limited precision c = decimal.getcontext().copy() c.prec = 3 # Create our constant pi = c.create_decimal('3.1415') # The constant value is rounded off print 'PI :', pi # The result of using the constant uses the global context print 'RESULT:', decimal.Decimal('2.01') * pi
这样一来,应用就可以区别于用户数据精度而另外选择常量值精度。
6. 线程
“全局”上下文实际上是线程本地上下文,所以完全可以使用不同的值分别配置各个线程。
import decimal import threading from Queue import PriorityQueue class Multiplier(threading.Thread): def __init__(self, a, b, prec, q): self.a = a self.b = b self.prec = prec self.q = q threading.Thread.__init__(self) def run(self): c = decimal.getcontext().copy() c.prec = self.prec decimal.setcontext(c) self.q.put( (self.prec, a * b) ) return a = decimal.Decimal('3.14') b = decimal.Decimal('1.234') # A PriorityQueue will return values sorted by precision, no matter # what order the threads finish. q = PriorityQueue() threads = [ Multiplier(a, b, i, q) for i in range(1, 6) ] for t in threads: t.start() for t in threads: t.join() for i in range(5): prec, value = q.get() print prec, '\t', value
这个例子使用指定的值创建一个新的上下文,然后安装到各个线程中。
总结
以上所述是小编给大家介绍的python中的decimal类型转换实例详解,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
稳了!魔兽国服回归的3条重磅消息!官宣时间再确认!
昨天有一位朋友在大神群里分享,自己亚服账号被封号之后居然弹出了国服的封号信息对话框。
这里面让他访问的是一个国服的战网网址,com.cn和后面的zh都非常明白地表明这就是国服战网。
而他在复制这个网址并且进行登录之后,确实是网易的网址,也就是我们熟悉的停服之后国服发布的暴雪游戏产品运营到期开放退款的说明。这是一件比较奇怪的事情,因为以前都没有出现这样的情况,现在突然提示跳转到国服战网的网址,是不是说明了简体中文客户端已经开始进行更新了呢?
更新日志
- 凤飞飞《我们的主题曲》飞跃制作[正版原抓WAV+CUE]
- 刘嘉亮《亮情歌2》[WAV+CUE][1G]
- 红馆40·谭咏麟《歌者恋歌浓情30年演唱会》3CD[低速原抓WAV+CUE][1.8G]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[320K/MP3][193.25MB]
- 【轻音乐】曼托凡尼乐团《精选辑》2CD.1998[FLAC+CUE整轨]
- 邝美云《心中有爱》1989年香港DMIJP版1MTO东芝首版[WAV+CUE]
- 群星《情叹-发烧女声DSD》天籁女声发烧碟[WAV+CUE]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[FLAC/分轨][748.03MB]
- 理想混蛋《Origin Sessions》[320K/MP3][37.47MB]
- 公馆青少年《我其实一点都不酷》[320K/MP3][78.78MB]
- 群星《情叹-发烧男声DSD》最值得珍藏的完美男声[WAV+CUE]
- 群星《国韵飘香·贵妃醉酒HQCD黑胶王》2CD[WAV]
- 卫兰《DAUGHTER》【低速原抓WAV+CUE】
- 公馆青少年《我其实一点都不酷》[FLAC/分轨][398.22MB]
- ZWEI《迟暮的花 (Explicit)》[320K/MP3][57.16MB]