复数可以写成 (A+B**i) 的常规形式,其中 A 是实部,B 是虚部,i 是虚数单位,满足 i2=−1;也可以写成极坐标下的指数形式 (R×e(P**i)),其中 R 是复数模,P 是辐角,i 是虚数单位,其等价于三角形式 R(cos(P)+isin(P))。
现给定两个复数的 R 和 P,要求输出两数乘积的常规形式。
输入格式:
输入在一行中依次给出两个复数的 R1, P1, R2, P2,数字间以空格分隔。
输出格式:
在一行中按照 A+Bi
的格式输出两数乘积的常规形式,实部和虚部均保留 2 位小数。注意:如果 B
是负数,则应该写成 A-|B|i
的形式。
输入样例:
2.3 3.5 5.2 0.4
输出样例:
-8.68-8.23i
题目解析:
本题其实就是纯粹的数学计算问题,计算公式如下:
r1(cos(p1)+isin(p1)) * r2(cos(p2)+isin(p2))
=r1*r2*[cosp1*cosp2-sinp1*sinp2 + (cosp1*sinp2+sinp1*cosp2)i]
所以只需要使用math.h中的函数进行计算就可以了,变量使用double类型即可。
#include<stdio.h>
#include<string.h>
#include<math.h>
#include<ctype.h>
#include<stdlib.h>
int main(){
double r1,p1,r2,p2;
scanf("%lf %lf %lf %lf",&r1,&p1,&r2,&p2);
double t1=r1*r2*(cos(p1)*cos(p2)-sin(p1)*sin(p2));
double t2=r1*r2*(cos(p1)*sin(p2)+sin(p1)*cos(p2));
if(t1==0){
printf("0.00");
}
else printf("%.2lf",t1);
if(t2>=0) printf("+%.2lfi",fabs(t2));
else printf("%.2lfi",t2);
return 0;
}
但是直接这么写会发现测试点2,3不通过,原因是 t2为0.004时,保留两位小数应该是0.00,本来应该是先保留小数再判断,我这里是直接用0.004判大于0才保留的小数,这里放一个测试点大家可以试一试
测试输入
0.00001 3 -0.0002 4
正确输出
0.00+0.00i
不过在保留小数时需要注意,如果直接使用double,可能会输出一个 -0.00 ,原因是因为 C语言中使用double类型时有一个+0还有一个-0,这里放出C语言中保留两位小数到变量的方式
方式一:使用sscanf和sprintf
char formatStr[100];
sprintf(formatStr,"%.2lf",t2);
sscanf(formatStr,"%lf",&t2);
sprintf(formatStr,"%.2lf",t1);
sscanf(formatStr,"%lf",&t1);
方式二:使用round
//保留二位小数
t1=round(t1*100)/100;
t2=round(t2*100)/100;
不过,也可以不预先保留成两位小数,直接将double数据与0.005比较大小也可以。其实就是手动进行四舍五入
if(fabs(t1)<0.005) t1=0.00
完整代码
#include<stdio.h>
#include<string.h>
#include<math.h>
#include<ctype.h>
#include<stdlib.h>
int main(){
double r1,p1,r2,p2;
scanf("%lf %lf %lf %lf",&r1,&p1,&r2,&p2);
double t1=r1*r2*(cos(p1)*cos(p2)-sin(p1)*sin(p2));
double t2=r1*r2*(cos(p1)*sin(p2)+sin(p1)*cos(p2));
//printf默认会四舍五入,但是0.004保留二位小数是+0,
//反之-0.004保留二位小数是-0,所以是有个+-0 的情况
//保留二位小数
t1=round(t1*100)/100;
t2=round(t2*100)/100;
if(t1==0){
printf("0.00");
}
else printf("%.2lf",t1);
if(t2>=0) printf("+%.2lfi",fabs(t2));
else printf("%.2lfi",t2);
return 0;
}