PAT 1051 复数乘法(15 分)

该博客详细介绍了PAT 1051复数乘法问题,包括输入输出格式,样例及解题思路。重点讨论了在处理复数乘法时,由于浮点数精度问题导致的误差,并提出了引入极小数eps进行误差修正的方法。博主推荐使用算法笔记中的解决方案,并指出在实际题目中调整eps值的重要性。
摘要由CSDN通过智能技术生成

1051 复数乘法(15 分)

复数可以写成 (A+Bi) 的常规形式,其中 A 是实部,B 是虚部,i 是虚数单位,满足 i2 i 2 ​​ =−1;也可以写成极坐标下的指数形式 ( R×ePi R × e P i ​​ ),其中 R 是复数模,P 是辐角,i 是虚数单位,其等价于三角形式 (R(cos(P)+isin(P)) ( R ( c o s ( P ) + i s i n ( P ) )
现给定两个复数的 R 和 P,要求输出两数乘积的常规形式。
输入格式:
输入在一行中依次给出两个复数的 R1 R ​ 1 ​ ​ , P1 P 1 ​​ , R2 R 2 ​​ , P2 P 2 ​​ ,数字间以空格分隔。
输出格式:
在一行中按照 A+Bi A + B i 的格式输出两数乘积的常规形式,实部和虚部均保留 2 位小数。注意:如果 B 是负数,则应该写成 A|B|i A − | B | i 的形式。
输入样例:

2.3 3.5 5.2 0.4

输出样例:

 -8.68-8.23i




解析

这题给了 R1×eP1i R 1 × e P 1 i ,​​ R2×eP2i R 2 × e P 2 i
可以用​​
这里写图片描述
代码里的R,P就是用这个公式算出来的。但是这样你只能得到监测点1的分,其他的分数呢?
在算法笔记里第一章以及给出了答案:
由于浮点数的存储不总是精确的,所以会给浮点数的比较带来极大的干扰.所以要引入一个极小数eps对这种误差进行修正
可以看<算法笔记>第二章第9节,或我的读书笔记看更多细节.
And。当你算出R,和P的值之后,
首先需要修正的是R,R可能在[-eps,+eps]之间浮动,但是其实是0.00,所以下面代码:

    if(Equ(R,0))
      printf("0.00");
    else
      printf("%.2lf",R);

在修正P,和上面的讨论一样。

    if(Equ(P,0.0))
      printf("+0.00i");
    else if(More(P,0))
      printf("+%.2lfi",P);
    else 
     printf("%.2lfi",P);

这题eps大于1e-5才能过.,但算法笔记中eps = 1e-8。第一次遇到浮点数比较的OJ题,还要继续观察eps设为什么值比较好.


算法笔记这本书真的很赞,如果你要考PAT,建议买一本看看。



code:

#include<stdio.h>
#include<math.h>
const double eps = 1e-5;
#define Equ(a,b) ((fabs((a)-(b)))<(eps))
#define More(a,b) (((a)-(b)) >(eps))
int main(){
    double R1,P1,R2,P2;
    scanf("%lf%lf%lf%lf",&R1,&P1,&R2,&P2);
    double R=R1*R2*(cos(P1+P2));
    double P=R1*R2*(sin(P1+P2));     // R+P
    if(Equ(R,0))
      printf("0.00");
    else
      printf("%.2lf",R);
    if(Equ(P,0.0))
      printf("+0.00i");
    else if(More(P,0))
      printf("+%.2lfi",P);
    else 
     printf("%.2lfi",P);
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值