引言

前一阵做了一个有理数四则混合运算的程序(详见:用C++实现的有理数(分数)四则混合运算计算器),以分数形式呈现运算结果。这次添加以循环小数形式呈现运算结果的功能。例如:

Please input a rational expression to calculate its value or input q to quit:

67+34*(23-78/(54*5))-2*(5/37-6.5789)
= 67 + 34 * ( 23 - 78 / ( 270 ))-2*(5/37-6.5789)
= 67 + 34 * ( 23 - 13/45 )-2*(5/37-6.5789)
= 67 + 34 * ( 1022/45 )-2*(5/37-6.5789)
= 67 + 34748/45 -2*(5/37-6.5789)
= 37763/45 -2*(5/37-6.5789)
= 37763/45 - 2 * ( 5/37 -6.5789)
= 37763/45 - 2 * ( 5/37 - 65789/10000 )
= 37763/45 - 2 * ( -2384193/370000 )
= 37763/45 - -2384193/185000
= 852[108737/1665000] {852.0653`075}

算法分析

由分数化循环小数不像由循环小数化分数那么简单明了。比如 0.2`142857 这个小数,它等于 0.2 加上 0.0`142857,用分数表示分别是 2/10 和 142857/9999990,化简即是 1/5 和 1/70,相加并化简得到分数运算结果:3/14。

现在考察由 3/14 如何反推出 0.2`142857。

1、3 / 14 = 0...3  商为0,得到小数形式结果的整数部分,余数不为0,继续

2、30 / 14 = 2...2  商为2,得到小数点后第1位小数,余数不为0,继续

3、20 / 14 = 1...6  商为1,得到小数点后第2位小数,余数不为0,继续

4、60 / 14 = 4...4 商为4,得到小数点后第3位小数,余数不为0,继续

5、40 / 14 = 2...12 商为2,得到小数点后第4位小数,余数不为0,继续

6、120 / 14 = 8...8 商为8,得到小数点后第5位小数,余数不为0,继续

7、80 / 14 = 5...10 商为5,得到小数点后第6位小数,余数不为0,继续

8、100 / 14 = 7...2 商为7,得到小数点后第7位小数,余数不为0,继续

9、20 / 14 = 1...6  商为1,得到小数点后第8位小数,余数不为0,继续

......

上述试商过程,当进行到第8步时,得到的余数是2,和第2步得到的余数相同,于是从第9步开始就会周而复始重复第3步到第8步的过程。说明已经找到了循环小数的循环体。

由于余数要小于指定的除数(即分数中的分母),上述过程必然会在有限的步数(小于除数)出现相同余数。

算法实现

在 SFraction 里增加 toDecimal 接口:

 1 struct SFraction
2 {
3 u64 numerator;
4 u64 denominator;
5 bool bNegative;
6
7 SFraction() {
8 numerator = 0;
9 denominator = 1;
10 bNegative = false;
11 }
12
13 std::string toStr(bool bFinal = false) const;
14 std::string toDecimalStr() const;
15 };

SFraction 的 toDecimal 接口实现如下:

 1 std::string SFraction::toDecimalStr() const
2 {
3 std::ostringstream oStream;
4 if (bNegative)
5 {
6 oStream << "-";
7 }
8 u64 quotient = numerator / denominator;
9 oStream << quotient;
10 u64 remainder = numerator % denominator;
11 if (remainder == 0)
12 {
13 return oStream.str();
14 }
15 oStream << ".";
16 u64 pos = 0;
17 u64 posMatched = 0;
18 std::map<u64, u64> mapRemainderPos;
19 std::vector<u64> vecQuotient;
20 while (true)
21 {
22 mapRemainderPos[remainder] = pos++;
23 remainder *= 10;
24 vecQuotient.push_back(remainder / denominator);
25 remainder = remainder % denominator;
26 if (remainder == 0)
27 break;
28 std::map<u64, u64>::iterator it = mapRemainderPos.find(remainder);
29 if (it != mapRemainderPos.end())
30 {
31 posMatched = it->second;
32 break;
33 }
34 }
35 if (remainder == 0)
36 {
37 for (size_t idx = 0; idx < vecQuotient.size(); ++idx)
38 oStream << vecQuotient[idx];
39 return oStream.str();
40 }
41 size_t idx = 0;
42 for (; idx < posMatched; ++idx)
43 oStream << vecQuotient[idx];
44 oStream << "`";
45 for (; idx < vecQuotient.size(); ++idx)
46 oStream << vecQuotient[idx];
47 return oStream.str();
48 }

然后对 SFraction 的 toStr 接口稍作调整,即可达到增加循环小数形式的呈现效果:

 1 std::string SFraction::toStr(bool bFinal) const
2 {
3 std::ostringstream oStream;
4 if (bNegative)
5 {
6 oStream << "-";
7 }
8 if (denominator == 1)
9 {
10 oStream << numerator;
11 return oStream.str();
12 }
13 if (!bFinal)
14 {
15 oStream << numerator << "/" << denominator;
16 return oStream.str();
17 }
18 if (numerator < denominator)
19 {
20 oStream << numerator << "/" << denominator << " {" << toDecimalStr() << "}";
21 return oStream.str();
22 }
23 u64 quotient = numerator / denominator;
24 u64 remainder = numerator % denominator;
25 oStream << quotient << "[" << remainder << "/" << denominator << "] {" << toDecimalStr() << "}";
26 return oStream.str();
27 }

完整代码文件提取位置

https://github.com/readalps/RationalCalculator

分数化循环小数C++实现的更多相关文章

  1. 洛谷P1530 分数化小数 Fractions to Decimals

    P1530 分数化小数 Fractions to Decimals 103通过 348提交 题目提供者该用户不存在 标签USACO 难度普及/提高- 提交  讨论  题解 最新讨论 暂时没有讨论 题目 ...

  2. [C++]2-5 分数化小数

    /* 分数化小数 输入正整数a,b,c,输出a/b的小数形式.精确到小数点后C位.a,b<=10^6,c<=10^6. 输入包含多组数据,结束标记为a=b=c=0 样例输入: 1 6 4 ...

  3. 分数化小数(decimal)

    分数化小数 ①我的程序 #include<iostream>using namespace std;int main(void){ int a,b,c,kase=0; while(scan ...

  4. YTU 1439: 2.4.5 Fractions to Decimals 分数化小数

    1439: 2.4.5 Fractions to Decimals 分数化小数 时间限制: 1 Sec  内存限制: 64 MB 提交: 194  解决: 13 题目描述 写一个程序,输入一个形如N/ ...

  5. Luogu P1530 分数化小数 Fractions to Decimals(模拟)

    P1530 分数化小数 Fractions to Decimals 题意 题目描述 写一个程序,输入一个形如\(N/D\)的分数(\(N\)是分子,\(D\)是分母),输出它的小数形式.如果小数有循环 ...

  6. [LeetCode] Fraction to Recurring Decimal 分数转循环小数

    Given two integers representing the numerator and denominator of a fraction, return the fraction in ...

  7. 【u237】分数化小数

    Time Limit: 1 second Memory Limit: 128 MB [问题描述] 写一个程序,输入一个形如N/D的分数(N是分子,D是分母),输出它的小数形式.如果小数有循环节的话,把 ...

  8. 【USACO 2.4.5】分数化小数

    [描述] 写一个程序,输入一个形如N/D的分数(N是分子,D是分母),输出它的小数形式. 如果小数有循环节的话,把循环节放在一对圆括号中. 例如, 1/3 =0.33333333 写成0.(3), 4 ...

  9. [LeetCode] 167. Fraction to Recurring Decimal 分数转循环小数

    Given two integers representing the numerator and denominator of a fraction, return the fraction in ...

随机推荐

  1. 第十六篇 -- SuperIO学习

    一.SuperIO 这次主要研究SuperIO读取以及控制风扇转速的问题. 参考文章:https://huchanghui123.github.io/Linux/Linux-Superio-CPU-F ...

  2. Pelles C编译时出现的“POLINK: fatal error: 拒绝访问”问题的一种可能成因

    在使用PellesC编译程序时,第一遍能正常编译执行,第二遍就无法编译,出现以下问题提示: Building NEWprogram2.exe. POLINK: fatal error: 拒绝访问. * ...

  3. 构建后端第5篇之---Idea 查看继承 实现关系图

    first question: how to show a class  children class :  move mousrmark to class name , Ctrl + H how t ...

  4. 【阅读笔记】Java核心技术卷一 #4.Chapter6

    6 接口.lambda 表达式与内部类 6.1 接口 6.1.1 接口概念 接口绝不能含有实例域:但在接口中可以定义常量,被自动设为 public static final 接口中的所有方法自动地属于 ...

  5. 图文实例解析,InnoDB 存储引擎中行锁的三种算法

    前文提到,对于 InnoDB 来说,随时都可以加锁(关于加锁的 SQL 语句这里就不说了,忘记的小伙伴可以翻一下上篇文章),但是并非随时都可以解锁.具体来说,InnoDB 采用的是两阶段锁定协议(tw ...

  6. 面试官疯狂问我:char和varchar的区别 怎么办?愣着干嘛?进来白嫖啊!

    MySQL的修仙之路,图文谈谈如何学MySQL.如何进阶!(已发布) 面前突击!33道数据库高频面试题,你值得拥有!(已发布) 大家常说的基数是什么?(已发布) 讲讲什么是慢查!如何监控?如何排查?( ...

  7. Arp欺骗和DNS投毒

    中间人攻击 ARP缓存攻击 ARP(Address Resolution Protocol,地址解析协议)是一个位于TCP/IP协议栈中的网络层,负责将某个IP地址解析成对应的MAC地址.简单来说,就 ...

  8. Java-Mybatis动态SQL整理

    XML映射器 SQL映射文件的几个顶级元素: cache - 该命名空间的缓存配置 cache-ref - 引用其他命名空间的缓存配置 resultMap - 描述如何从数据库结果集中加载对象 sql ...

  9. msfvenom简介

    写此文是因为网上资料杂乱,不方便查阅,辣眼睛 测试免杀的时候刚好用到这个功能,顺便写一下(0202年靠msfvenom生成的纯原生payload可以宣告死亡了,如果有查不出来的杀软可以退群了,这也叫杀 ...

  10. MATLAB—常用控制流

    文章目录 一.MATLAB控制流与C语言的区别 二.if-else-end 判断 1.使用方法 2.例题 三.switch-case 分支 1.使用方法 2.例题 四.for.while循环 1.使用 ...