解决Python中的modf()函数取小数部分不准确问题


Posted in Python onMay 28, 2021

使用math.modf()对一个浮点数进行拆分时经常会遇到如下情况

如下

import math
print(math.modf(2.4)) # 输出 (0.3999999999999999, 2.0)

我们会发现对2.4进行拆分得到的小数并不是0.4,这是因为什么呢?

这是因为计算机采用的是二进制代码,而二进制代码由于计算上的误差无法准确表示某些十进制数的小数部分。

下面我们具体来讲一下。

我们知道一个十进制数转化为二进制数需要分为两部分进行计算:整数部分和小数部分。

整数部分采用“除二取余法”。将这个整数除以2,得到它的余数,然后将商再除以3,直到商为0为止,并将各个得到的余数按照相反的顺序进行排列。

小数部分采用“乘2取整法”,将这个小数乘2,将新得到的数的整数部分取出,再用2乘余下的小数部分,如此往复直到乘积中的小数部分为0或者达到要求的精度为止。并将取出的整数部分按照取出的先后顺序从前到后排列。

123/2=61...1
 61/2=30...1
 30/2=15...0
 15/2=7...1
 7/2=3...1
 3/2=1...1
 1/2=0...1
 (123)10=(1111011)2
0.4*2=0.8...0
0.8*2=1.6...1
0.6*2=1.2...1
0.2*2=0.4...0
0.4*2=0.8...0
0.8*2=1.6...1
0.6*2=1.2...1
(0.4)10=(0.011001100110011001100110...)2
(123.4)2=(1111011.011001100110011...)2

十进制的0.4转化为二进制时会出现重复循环“0110”的情况,但是目前计算机存储浮点数是按照电器和电子工程师协会的标准(IEEE754浮点数存储格式标准)来进行存储的。

IEEE规定单精度浮点最多存储32位(4个字节):

31位是符号位。1表示该数为负,0为正

30~23是指数位(-128-127)

22~0是尾数位,尾数的编码一般是源码和补码

IEEE标准从逻辑上用三元组{S,E,M}表示一个数,如图所示:

解决Python中的modf()函数取小数部分不准确问题

也就是说上面将0.4转换出的二进制代码,我们只能存储23位,即使数据类型为double,也只能存储52位,这样大家便能看出问题出现的原因了。23位的数据显然无法完整表示0.4的二进制数据,于是误差产生了。

那所有的十进制小数都无法完整表示吗?不是的,只要小数部分乘上2最终小数部分能够得到0就不会出现这种问题,比如0.5,0.75。

import math
print(math.modf(1.5))  # 输出 (0.5,1.0)
0.5*2=1...1
(0.5)10=(0.1)2
0.75*2=1.5...1
0.5*2=1...1
(0.75)10=(0.11)2

补充:Python中“取整”的各种问题

一、初衷:

有时候我们分页展示数据的时候,需要计算页数。一般都是向上取整,例如counts=205 pageCouts=20 ,pages= 11 页。

一般的除法只是取整数部分,达不到要求。

二、方法:

1、通用除法:

UP(A/B) = int((A+B-1)/B)

取临界值,计算下A+B-1的范围就OK.

2 、Python除法:

首先要说的是python中的除法运算,

当使用x/y形式进行除法运算时,那么会进行所谓的true除法,比如2.0/3的结果是 0.66666666666666663。

另外一种除法是采用x//y的形式(向下取整),那么这里采用的是所谓floor除法,即得到不大于结果的最大整数值,这个运算时与操作数无关的。比如2//3的结果是0,-2//3的结果是-1,-2.0//3的结果是-1.0。

在python 3.0中,x/y将只执行true除法,而与操作数无关;x//y则执行floor除法。

Python运算向上取整方法:(A+B-1)/B

3、Python match.ceil函数 np.ceil函数

ceil(x)函数是向上取整,即取大于等于x的最接近整数。

import math 
math.ceil(float(205)/20) 
import numpy as np
np.ceil(float(205)/20)

Python 向上取整的算法

1、一般使用floor除法 (np.floor()或者math.floor())

import numpy as np 
n = np.array([-1.7, -2.5, -0.2, 0.6, 1.2, 2.7, 11]) 
floor = np.floor(n)
print(floor)  # [ -2.  -3.  -1.   0.   1.   2.  11.]

2、一般除法/

A=100
B=16
c=100//16
(c=6)

3、round()四舍五入函数。

np.around 返回四舍五入后的值,可指定精度。

around(a, decimals=0, out=None)

a 输入数组

decimals 要舍入的小数位数。 默认值为0。 如果为负,整数将四舍五入到小数点左侧的位置

import numpy as np 
n = np.array([-0.746, 4.6, 9.4, 7.447, 10.455, 11.555])
 
around1 = np.around(n)
print(around1)  # [ -1.   5.   9.   7.  10.  12.]
 
around2 = np.around(n, decimals=1)
print(around2)  # [ -0.7   4.6   9.4   7.4  10.5  11.6]
 
around3 = np.around(n, decimals=-1)
print(around3)  # [ -0.   0.  10.  10.  10.  10.]

Python 分别取整的算法

math模块中的 modf()方法

将整数部分和小数部分分别取出,可以使用math模块中的 modf()方法

例如:

>>> math.modf(4.25)
(0.25, 4.0)
>>> math.modf(4.33)

以上为个人经验,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
详解Python中find()方法的使用
May 18 Python
在Mac OS上搭建Python的开发环境
Dec 24 Python
Django Highcharts制作图表
Aug 27 Python
python网络爬虫学习笔记(1)
Apr 09 Python
Python @property使用方法解析
Sep 17 Python
pygame实现贪吃蛇游戏(上)
Oct 29 Python
pytorch中的自定义数据处理详解
Jan 06 Python
Jupyter notebook运行Spark+Scala教程
Apr 10 Python
通过python调用adb命令对App进行性能测试方式
Apr 23 Python
python对一个数向上取整的实例方法
Jun 18 Python
PyCharm配置anaconda环境的步骤详解
Jul 31 Python
python实战之90行代码写个猜数字游戏
Apr 22 Python
利用Python+OpenCV三步去除水印
python实现自定义日志的具体方法
May 28 #Python
python 爬取京东指定商品评论并进行情感分析
python b站视频下载的五种版本
May 27 #Python
教你怎么用python selenium实现自动化测试
Python Django框架介绍之模板标签及模板的继承
May 27 #Python
python 算法题——快乐数的多种解法
May 27 #Python
You might like
一个PHP模板,主要想体现一下思路
2006/12/25 PHP
php session 检测和注销
2009/03/16 PHP
php中filter函数验证、过滤用户输入的数据
2014/01/13 PHP
php将图片保存为不同尺寸图片的图片类实例
2015/03/30 PHP
Laravel 5框架学习之日期,Mutator 和 Scope
2015/04/08 PHP
Yii操作数据库实现动态获取表名的方法
2016/03/29 PHP
Laravel5.1框架路由分组用法实例分析
2020/01/04 PHP
Yii 实现数据加密和解密
2021/03/09 PHP
Js sort排序使用方法
2011/10/17 Javascript
js图片自动轮播代码分享(js图片轮播)
2014/05/06 Javascript
IE6-IE9中tbody的innerHTML不能赋值的解决方法
2014/06/05 Javascript
javascript中setTimeout使用指南
2015/07/26 Javascript
jQuery跨域问题解决方案
2015/08/03 Javascript
jQuery定义插件的方法
2015/12/18 Javascript
jQuery简单动画变换效果实例分析
2016/07/04 Javascript
jQuery焦点图轮播效果实现方法
2016/12/19 Javascript
node.js + socket.io 实现点对点随机匹配聊天
2017/06/30 Javascript
基于JavaScript实现表格滚动分页
2017/11/22 Javascript
详解如何在Vue项目中发送jsonp请求
2019/10/25 Javascript
js实现选项卡效果
2020/03/07 Javascript
JS实现简单贪吃蛇小游戏
2020/10/28 Javascript
pandas DataFrame 警告(SettingWithCopyWarning)的解决
2019/07/23 Python
Pytorch中Tensor与各种图像格式的相互转化详解
2019/12/26 Python
Python求区间正整数内所有素数之和的方法实例
2020/10/13 Python
python 爬虫请求模块requests详解
2020/12/04 Python
html5声频audio和视频video等新特性详细说明
2012/12/26 HTML / CSS
jQuery treeview树形结构应用
2021/03/24 jQuery
职业生涯规划书范文
2014/03/10 职场文书
幸福中国演讲稿
2014/09/12 职场文书
毕业生个人总结
2015/02/28 职场文书
拾金不昧通报表扬范文
2015/05/05 职场文书
红白喜事主持词
2015/07/06 职场文书
校园之声广播稿
2015/08/18 职场文书
使用python向MongoDB插入时间字段的操作
2021/05/18 Python
Anaconda安装pytorch及配置PyCharm 2021环境
2021/06/04 Python
用Python实现屏幕截图详解
2022/01/22 Python