本题要求编写程序,计算 2 个有理数的和、差、积、商。

输入格式:

输入在一行中按照 a1/b1 a2/b2 的格式给出两个分数形式的有理数,其中分子和分母全是整型范围内的整数,负号只可能出现在分子前,分母不为 0。

输出格式:

分别在 4 行中按照 有理数1 运算符 有理数2 = 结果 的格式顺序输出 2 个有理数的和、差、积、商。注意输出的每个有理数必须是该有理数的最简形式 k a/b,其中 k 是整数部分,a/b 是最简分数部分;若为负数,则须加括号;若除法分母为 0,则输出 Inf。题目保证正确的输出中没有超过整型范围的整数。

输入样例 1:

  1. 2/3 -4/2

输出样例 1:

  1. 2/3 + (-2) = (-1 1/3)
  2. 2/3 - (-2) = 2 2/3
  3. 2/3 * (-2) = (-1 1/3)
  4. 2/3 / (-2) = (-1/3)

输入样例 2:

  1. 5/3 0/6

输出样例 2:

  1. 1 2/3 + 0 = 1 2/3
  2. 1 2/3 - 0 = 1 2/3
  3. 1 2/3 * 0 = 0
  4. 1 2/3 / 0 = Inf

思路

  • 用结构体来模拟分数运算
  • 两个int型变量相乘,结果可能超过int型承受的范围。所有我的分子分母均采用long long int
  • 必须在每一步加法后都进行约分。如果等全部加完后才约分,则会溢出

有些例子容易出错:

  1. 2
  2. -1/2 2/4
  3. // Output:
  4. 0
  1. 2
  2. 1/3 -1/2
  3. // Output:
  4. -1/6

代码

  1. #include<cstdio>
  2. #include<algorithm>
  3. using namespace std;
  4. struct Fraction {
  5. long long int up;
  6. long long int down;
  7. };
  8. long long int gcd(long long int, long long int);
  9. Fraction reduction(Fraction result); /* 对分数进行约分 */
  10. Fraction add(Fraction f1, Fraction f2);
  11. Fraction minus(Fraction f1, Fraction f2);
  12. Fraction multiply(Fraction f1, Fraction f2);
  13. Fraction divide(Fraction f1, Fraction f2);
  14. void display(Fraction result); /* 展示Fraction对象 */
  15. int main() {
  16. Fraction number1, number2, result;
  17. scanf("%lld/%lld %lld/%lld", &number1.up, &number1.down, &number2.up, &number2.down);
  18. /** 展示加法 */
  19. display(number1);
  20. printf(" + ");
  21. display(number2);
  22. printf(" = ");
  23. display(add(number1, number2));
  24. printf("\n");
  25. /** 展示减法 */
  26. display(number1);
  27. printf(" - ");
  28. display(number2);
  29. printf(" = ");
  30. display(minus(number1, number2));
  31. printf("\n");
  32. /** 展示乘法 */
  33. display(number1);
  34. printf(" * ");
  35. display(number2);
  36. printf(" = ");
  37. display(multiply(number1, number2));
  38. printf("\n");
  39. /** 展示除法 */
  40. display(number1);
  41. printf(" / ");
  42. display(number2);
  43. printf(" = ");
  44. if(number2.up == 0) {
  45. printf("Inf\n");
  46. }
  47. else {
  48. display(divide(number1, number2));
  49. printf("\n");
  50. }
  51. return 0;
  52. }
  53. /***********************函数实现**********************************/
  54. long long int gcd(long long int m, long long int n) {
  55. if(n == 0)
  56. return m;
  57. else {
  58. int remainder;
  59. while(n) { /* 辗转相除法求最大公因数 */
  60. remainder = m % n;
  61. m = n;
  62. n = remainder;
  63. }
  64. return m;
  65. }
  66. }
  67. Fraction reduction(Fraction result) {
  68. if(result.down < 0) { /* 如果分数是负的,我们统一把负号放到分子 */
  69. result.up = -result.up;
  70. result.down = -result.down;
  71. }
  72. if(result.up == 0) { /* 如果分子为0 */
  73. result.down = 1; /* 令分母为1 */
  74. }
  75. else { /* 如果分子不为0 */
  76. /* 找分子分母的最大公因数 */
  77. int factor = gcd(abs(result.up), abs(result.down));
  78. result.up /= factor; /* 约掉最大公因数 */
  79. result.down /= factor;
  80. }
  81. return result;
  82. }
  83. Fraction add(Fraction f1, Fraction f2) {
  84. Fraction result;
  85. result.up = (f1.up * f2.down) + (f2.up * f1.down);
  86. result.down = f1.down * f2.down;
  87. return reduction(result);
  88. }
  89. Fraction minus(Fraction f1, Fraction f2) {
  90. Fraction result;
  91. result.up = (f1.up * f2.down) - (f2.up * f1.down);
  92. result.down = f1.down * f2.down;
  93. return reduction(result);
  94. }
  95. Fraction multiply(Fraction f1, Fraction f2) {
  96. Fraction result;
  97. result.up = f1.up * f2.up;
  98. result.down = f1.down * f2.down;
  99. return reduction(result);
  100. }
  101. Fraction divide(Fraction f1, Fraction f2) {
  102. Fraction result;
  103. result.up = f1.up * f2.down;
  104. result.down = f1.down * f2.up;
  105. return reduction(result);
  106. }
  107. void display(Fraction result) {
  108. result = reduction(result);
  109. if(result.up < 0)
  110. printf("(");
  111. if(result.up == 0) {
  112. printf("0");
  113. }
  114. else if(result.down == 1) { /* 分母为1,是一个整数 */
  115. printf("%lld", result.up);
  116. }
  117. else if(abs(result.up) > abs(result.down)) { /* 假分数 */
  118. printf("%d %d/%d", result.up / result.down, abs(result.up) % result.down, result.down);
  119. }
  120. else { /* 真分数 */
  121. printf("%d/%d", result.up, result.down);
  122. }
  123. if(result.up < 0)
  124. printf(")");
  125. }