Python算法100例-4.8 计算分数的精确值

完整源代码项目地址,关注博主私信'源代码'后可获取

1.问题描述

使用数组精确计算M/N(0<M<N≤100)的值。假如M/N是无限循环小数,则计算并输出它的第一循环节,同时要求输出循环节的起止位置(小数位的序号)。

2.问题分析

由于计算机字长的限制,常规的浮点运算都有精度限制,为了得到高精度的计算结果,就必须自行设计实现方法。

为了实现高精度的计算,可将商存放在一维数组中,数组中的每个元素存放一位十进制数,即商的第一位存放在第一个元素中,商的第二位存放在第二个元素中,以此类推。这样就可以使用数组来表示一个高精度的计算结果。

进行除法运算时可以模拟人工操作,即每次求出商的第一位后,将余数乘以10,再计算商的下一位,重复以上过程,当某次计算后的余数为0时,表示M/N为有限不循环小数。某次计算后的余数与前面的某个余数相同时,则M/N为无限循环小数,从该余数第一次出现之后所求得的各位数就是小数的循环节。

程序具体实现时,采用了数组和其他一些技巧来保存除法运算所得到的余数和商的各位数。

3.算法分析

在运算过程中,每次得到的余数都要与前面运算过程产生的余数进行比较,看否已经出现过,故余数及商都要存储在数组中。分别定义两个数组remainder[101]和quotient[101],用来存放运算过程中每一步的余数,及得到的每一位商。

被除数、除数分别用变量m、n存储,由题意知m<n。数组元素remainder[m]的下标表示运算过程中得到的余数,remainder[m]=i语句的作用是:记录余数m所存储的位置i,即小数点后的位置。quotient[i]=m/n把第i次相除得到的商存储到数组中下标为i的位置。因每次除完之后,得到的余数肯定比除数n小,所以在下一次进行相除之前余数应先乘以10再运算。代码如下:

remainder[m] = i                # m:得到的余数;remainder[m]:该余数对应的商的位数
m *= 10                                 # 余数扩大10倍
quotient[i] = m // n    # 商
m = m % n                               # 求余数

对得到的余数m进行判断,看m的值是否为0,若为0则说明此分数是有限循环小数,结束循环。若m的值不为0,则要判断此余数是否已经出现过(定义时将数组remainder中元素的值全部初始化为0,若remainder[m]的值为0则说明此余数没有出现过,若值不为0则说明前面已出现过相同的余数,此分数为无限循环小数),若没有出现过则继续相除,若已经出现过则说明分数为无限循环小数,第一个循环节的起始位置为数组quotient[1]的结束位置,数组元素remainder[m]对应的下标i。代码如下:

if m == 0:                                              # 余数为0则表示是有限小数
    j = 1
    while j <= i:
        print("%d" %quotient[j], end="")                    # 输出商
        j += 1
    break                                                                                   # 退出循环

if remainder[m] != 0:                           # 若该余数对应的位在前面已经出现过
    j = 1
    while j <= i:
        print("%d" %quotient[j], end="")                    # 输出循环小数
        j += 1
    print("\n分数的第一个循环节:%d" %remainder[m])
    print("循环节的起始位置为:%d" %i)
    break

4.完整的程序

根据上面的分析,编写程序如下:

# 计算分数精确值

if __name__ == "__main__":
    remainder = [0]*101                             # remainder存放除法的余数
    quotient = [0]*101                              # quotient 依次存放商的每一位
    print("请输入分子与分母大于0,小于等于100的分数")
    m = int(input("分子m = "))                        # 分子
    n = int(input("分母n = "))                        # 分母
    print("输入的分数为:%d/%d" %(m, n))
    print("%d/%d 的准确值是:0." %(m, n), end="")
    for i in range(1, 101):                         # i商的位数
        remainder[m] = i    # m:得到的余数;remainder[m]:该余数对应的商的位数
        m *= 10                                                     # 余数扩大10倍
        quotient[i] = m // n                        # 商
        m = m % n                                                   # 求余数
        if m == 0:                                                  # 余数为0则表示是有限小数
            j = 1
            while j <= i:
                print("%d" %quotient[j], end="")            # 输出商
                j += 1
            break   # 退出循环

        if remainder[m] != 0:                       # 若该余数对应的位在前面已经出现过
            j = 1
            while j <= i:
                print("%d" %quotient[j], end="")    # 输出循环小数
                j += 1
            print("\n分数的第一个循环节:%d" %remainder[m])
            print("循环节的起始位置为:%d" %i)
            break


请输入分子与分母大于0,小于等于100的分数
输入的分数为:1/6
1/6 的准确值是:0.16
分数的第一个循环节:2
循环节的起始位置为:2
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

飘逸高铁侠

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值