c++作业题sin公式

今日 有一位同样读大一的朋友向我求助有关c++的作业题 他说他的程序逻辑正确 但是结果的精度不对

题目如下:


这是一道看起来十分简单的作业题 我按照要求快速地写了一个版本 不出所料 一样遇到了精度问题

为什么会出现这种问题?

首先 计算机中的浮点数是不够精确的 这是为了运算速度所做的牺牲

在我写的代码中 使用的是double 为了使结果精确 要能尽量减少不必要的计算过程

而我用了cmath头文件中的pow函数 这使得double会被计算很多次导致结果精度降低

于是我决定写一个无pow版本 从公式出发 思路如下:

原式化简:

sin(x) = x/1! - x^3/(1!*2*3) + x^5/(3!*4*5) - x^7/(5!*6*7) ...

拆分:

sin(x)= x/1!
	- x^3/(1!*2*3)
	+ x^5/(3!*4*5)
	- x^7/(5!*6*7)
	...

拆分之后我们可以尝试用一个变量去替换其中的数字:

sin(x)= x/1!
[i=1] - x^3/(i!*(i+1)*(i+2))
[i=3] + x^5/(i!*(i+1)*(i+2))
[i=5] - x^7/(i!*(i+1)*(i+2))
[i=n] ...

引入一个变量p 替换幂:

p=x
sin(x)= p/1!
[i=1,p=p*x*x] - p/(i!*(i+1)*(i+2))
[i=3,p=p*x*x] + p/(i!*(i+1)*(i+2))
[i=5,p=p*x*x] - p/(i!*(i+1)*(i+2))
[i=n,p=p*x*x] ...

再引入一个变量f 替换阶乘:

f=1
p=x
sin(x)= p/f
[i=1,p=p*x*x,f=f*(i+1)*(i+2)] - p/f
[i=3,p=p*x*x,f=f*(i+1)*(i+2)] + p/f
[i=5,p=p*x*x,f=f*(i+1)*(i+2)] - p/f
[i=n,p=p*x*x,f=f*(i+1)*(i+2)] ...

再引入一个变量n 替换其中的负号:

n=1
f=1
p=x
sin(x)= p/f
[i=1,p=p*x*x,f=f*(i+1)*(i+2),n=-n] + n*p/f
[i=3,p=p*x*x,f=f*(i+1)*(i+2),n=-n] + n*p/f
[i=5,p=p*x*x,f=f*(i+1)*(i+2),n=-n] + n*p/f
[i=n,p=p*x*x,f=f*(i+1)*(i+2),n=-n] ...

现在每项的式子完全相同了 只需将其转换成对应的程序
我们要考虑结束条件 即当其中一项的绝对值小于1.0e-7时 结束这个循环

最终答案 :

 1 double x;
 2 cout << "请输入角度值:";
 3 cin >> x;
 4 
 5 x = x * 3.141592654 / 180;
 6 
 7 double n = 1;
 8 double f = 1;
 9 double p = x;
10 double i = -1;
11 
12 double term = 0;
13 double result = p / f;
14 
15 while (1)
16 {
17     i += 2;
18     p = p * x * x;
19     f = f * (i + 1) * (i + 2);
20     n = -n;
21     term = p / f;
22     if (term < 1.0e-7)
23         break;
24     result = result + n * term;
25 }
26 cout << "对应的sin值为:" << result << endl;

结果正确:

 

再看看朋友的代码:

 1 double angel,term,sum=0,i=1,multi=1,sign=-1;
 2 const double pi=3.141592654;
 3 cout<<"请输入角度值:";
 4 cin>>angel;
 5 angel=angel*pi/180;
 6 term=angel;
 7 while(term>=1.0e-7||term<=-1.0e-7)
 8 {
 9     sum+=term;
10     angel*=angel*angel;
11     multi*=(i+1)*(i+2);
12     term=sign*angel/multi;
13     sign*=-1;
14     i+=2;
15 }
16 cout<<"对应的sin值为:"<<sum<<endl;

结果:

思路大致和我相同 但是其中一步 angel*=angel*angel; 不对
在我的代码中 这一步对应的是 p = p * x * x; 其中x是不变的 他代码的这一部分并不等效于sin公式

END

posted @ 2019-10-27 10:10  redstone_torch  阅读(843)  评论(0编辑  收藏  举报