`
yhz61010
  • 浏览: 562504 次
  • 来自: -
博客专栏
63c13ecc-ef01-31cf-984e-de461c7dfde8
libgdx 游戏开发
浏览量:12273
社区版块
存档分类
最新评论

(转)一个让人发狂的PI求解C程序

阅读更多
    记得是大四的时候,我第一次看到了这个程序。那段时间,我对各种稀奇的C程序很感兴趣,也写了很多程序,但是当时看到这个程序时,着实是吓了一大跳。
    今天无意着又想起这个程序,所以上网找了找。觉得找到了一个不错文章解析这个程序的。现在转载过来与大家再次分析。

    在网上也找到个另类变态的类似程序,但是我没有执行成功。不过还是把地址贴出来与大家分享下,还请高手指教如何运行。 :)

    原文地址:
    1. http://blogger.org.cn/blog/more.asp?name=njucs&id=10151

    2. http://blog.est.im/archives/877
    详见三楼的回复。
    1988年IOCCC的Best layout奖,作者Merlyn LeRoy (Brian Westley).
    http://ioccc.org/years.html#1988_westley

    注:以上两个地址中,说的都是同一个程序。

本文转载自以下地址:
http://hi.baidu.com/jumbo/blog/item/155cb01b13cfba198618bf89.html

#include <stdio.h>

long a=10000, b, c=2800, d, e, f[2801], g;

main(){ 
for(;b-c;) f[b++]=a/5;
for(;d=0,g=c*2;c-=14,printf("%.4d",e+d/a),e=d%a)
for(b=c; d+=f[b]*a, f[b]=d%--g, d/=g--, --b; d*=b);
}


    简短的4行代码,就可以精确计算机出800位的PI(圆周率)值。
实在太震撼人心了。这样的程序也能运行,竟然还能完成这样让人难以置信的任务,真是太神了。

    这是某一年The International Obfuscated C Code Contest(国际模糊C代码大赛)上的获奖作品(努力了,但是没有找到一个确切的时间)。这是属于C大师的盛会,因为这是一件极具挑战的活儿。

    一、源程序
    本文分析下面这个很流行的计算PI的小程序。下面这个程序初看起来似乎摸不到头脑,
不过不用担心,当你读完本文的时候就能够基本读懂它了。
    程序一:很牛的计算Pi的程序
int a=10000,b,c=2800,d,e,f[2801],g;
main() {
for(;b-c;)
    f[b++]=a/5;
for(;d=0,g=c*2;c -=14,printf("%.4d",e+d/a),e=d%a)
    for(b=c; d+=f[b]*a,f[b]=d%--g,d/=g--,--b; d*=b);
}

    二、数学公式
    数学家们研究了数不清的方法来计算PI,这个程序所用的公式如下:
          1          2          3                    k
pi = 2 + --- * (2 + --- * (2 + --- * (2 + ...  (2 + ---- * (2 + ... ))...)))

          3          5          7                   2k+1
    至于这个公式为什么能够计算出PI,已经超出了本文的能力范围。
    下面要做的事情就是要分析清楚程序是如何实现这个公式的。
    我们先来验证一下这个公式:
    程序二:Pi公式验证程序
#include "stdio.h"
void main()
{
   float pi=2;
   int  i;
   for(i=100;i>=1;i--)
      pi=pi*(float)i/(2*i+1)+2;
   printf("%f\n",pi);
   getchar();
}

    上面这个程序的结果是3.141593。

    三、程序展开
    在正式分析程序之前,我们需要对程序一进行一下展开。我们可以看出程序一都是使用
for循环来完成计算的,这样做虽然可以使得程序短小,但是却很难读懂。根据for循环
的运行顺序,我们可以把它展开为如下while循环的程序:
    程序三:for转换为while之后的程序
#include <stdio.h>
int a=10000,b,c=2800,d,e,f[2801],g;
main() {
int i;
for(i=0;i<c;i++)
     f[i]=a/5;
while(c!=0)
     {
         d=0;
         g=c*2;
         b=c;
         while(1)
            {
                d=d+f[b]*a;
                g--;
                f[b]=d%g;
                d=d/g;
                g--;
                b--;
                if(b==0) break;
                d=d*b;
            }
         c=c-14;
         printf("%.4d",e+d/a);
         e=d%a;
    }
}

注:for([1];[2];[3]) {[4];}的运行顺序是[1],[2],[4],[3]。如果有逗号操作符,例如:d=0,g=c*2,则先运行d=0,
然后运行g=c*2,并且最终的结果是最后一个表达式的值,也就是这里的c*2。
   
    下面我们就针对展开后的程序来分析。
    四、程序分析
    要想计算出无限精度的PI,我们需要上述的迭代公式运行无数次,并且其中每个分数也
是完全精确的,这在计算机中自然是无法实现的。那么基本实现思想就是迭代足够多次
,并且每个分数也足够精确,这样就能够计算出PI的前n位来。上面这个程序计算800位
,迭代公式一共迭代2800次。

int a=10000,b,c=2800,d,e,f[2801],g;

    这句话中的2800就是迭代次数。
    由于float或者double的精度远远不够,因此程序中使用整数类型(实际是长整型),分段运算(每次计算4位)。我们可以看到输出语句 printf("%.4d",e+d/a); 其中%.4就是把计算出来的4位输出,我们看到c每次减少14( c=c-14;),而c的初始大小为2800,因
此一共就分了200段运算,并且每次输出4位,所以一共输出了800位。
    由于使用整型数运算,因此有必要乘上一个系数,在这个程序中系数为1000,也就是说
,公式如下:
               1          2          3                    k
1000*pi = 2k+ --- * (2k+ --- * (2k+ --- * (2k+ ...  (2k+ ---- * (2k+ ...))...)))
               3          5          7                   2k+1
    这里的2k表示2000,也就是f[2801]数组初始化以后的数据,a=10000,a/5=2000,所以下面
的程序把f中的每个元素都赋值为2000:
for(i=0;i<c;i++)
     f[i]=a/5;
    你可能会觉得奇怪,为什么这里要把一个常数储存到数组中去,请继续往下看。
我们先来跟踪一下程序的运行:
while(c!=0)  假设这是第一次运行,c=2800,为迭代次数
     {
         d=0;
         g=c*2;  这里的g是用来做k/(2k+1)中的分子
         b=c;    这里的b是用来做k/(2k+1)中的分子
         while(1)
         while(1)
            {
                d=d+f[b]*a; f中的所有的值都为2000,这里在计算时又把系数扩大了a=10000倍。这样做的目的稍候介绍,你可以看到输出的时候是d/a,所以这不影                        计算
                g--;
                f[b]=d%g; 先不管这一行
                d=d/g;   第一次运行的g为2*2799+1,你可以看到g做了分母
                g--;
                b--;
                if(b==0) break;
                d=d*b; 这里的b为2799,可以看到b做了分子。
            }
         c=c-14;
         printf("%.4d",e+d/a);
         e=d%a;
    }

    只需要粗略的看看上面的程序,我们就大概知道它的确是使用的那个迭代公式来计算Pi
的了,不过不知道到现在为止你是否明白了f数组的用处。如果没有明白,请继续阅读。

    d=d/g,这一行的目的是除以2k+1,我们知道之所以程序无法精确计算的原因就是这个除法。即使用浮点数,答案也是不够精确的,因此直接用来计算800位的Pi是不可能的。那
么不精确的成分在哪里?很明显:就是那个余数d%g。程序用f数组把这个误差储存起来
,再下次计算的时候使用。现在你也应该知道为什么d=d+f[b]*a;中间需要乘上a了吧。
把分子扩大之后,才好把误差精确的算出来。
    d如果不乘10000这个系数,则其值为2000,那么运行d=d/g;则是2000/(2*2799+1),这种整数的除法答案为0,根本无法迭代下去了。
    现在我们知道程序就是把余数储存起来,作为下次迭代的时候的参数,那么为什么这么
做就可以使得下次迭代出来的结果为接下来的4位数呢?
    这实际上和我们在纸上作除法很类似:
       0142
    /——------
7 /   1
       10
        7
        7
---------------
        30
    28
---------------
         20
         14
---------------
          60
     .....
    我们可以发现,在做除法的时候,我们通常把余数扩大之后再来计算,f中既然储存的是
余数,而f[b]*a;则正好把这个余数扩大了a倍,然后如此循环下去,可以计算到任意精
度。
    这里要说明的是,事实上每次计算出来的d并不一定只有4位数,例如第一次计算的时候
,d的值为31415926,输出4位时候,把低四位的值储存在e中间,e=d%a,也就是5926。
    最后,这个c=c-14不太好理解。事实上没有这条语句,程序计算出来的仍然正确。只是
因为如果迭代2800次,无论分数如何精确,最后Pi的精度只能够达到800。
你可以把程序改为如下形式尝试一下:
for(i=0;i<800;i++)
     {
     {
         d=0;
         g=c*2;
         b=c;
         while(1)
            {
                d=d+f[b]*a;
                g--;
                f[b]=d%g;
                d=d/g;
                g--;
                b--;
                if(b==0) break;
                d=d*b;
            }
        // c=c-14; 不要这句话。
         printf("%.4d",e+d/a);
         e=d%a;
    }
    最后的答案仍然正确。
    不过我们可以看到内循环的次数是c次,也就是说每次迭代计算c次。而每次计算后续位
数的时候,迭代次数减少14,而不影响精度。
分享到:
评论

相关推荐

    找乐儿-百度贴吧发狂 v1.0

    1. **BaiduTiebaFaKuang.exe**:这是主应用程序文件,通常是一个可执行程序,用户可以通过运行这个文件来启动"找乐儿-百度贴吧发狂 v1.0"软件。 2. **chinaz.com说明.txt**:这可能是一个来自知名网站Chinaz(站长...

    三子连珠游戏VC++程序

    编一棋盘游戏程序,可以选择人机对战和双人对战两种游戏模式,而人机模式下可以选择三种电脑等级,即简单的电脑、中等的电脑和令人发狂的电脑,电脑智能化程度随等级提高而提高。下棋双方分别以字符X和字符O为棋子。...

    c语言教程-一本c教程

    c语言教程

    matlabSilulink程序源代码与模型

    `mySfuntmpl.m`可能是一个自定义S-Function模板,提供了一个基础框架,帮助开发者快速创建新的S-Functions。 接下来是两个MATLAB脚本文件:`ch7example12prog1.m`和`ch7example6prog1.m`。这些是MATLAB脚本,通常...

    方便快捷通讯录挺使用的一个东西

    【标题】:“方便快捷通讯录挺使用的一个东西” 在当今数字化时代,通讯录软件成为了我们日常生活和工作中不可或缺的工具。标题中的“方便快捷通讯录”指的是这样一类应用,它能够帮助用户快速、高效地管理联系人...

    全套人教版九年级英语Unit 11同步练习题及答案13精选.doc

    第三句描绘了一个人的情绪状态,而第四句则用"neither…nor…"结构表达了双重否定。第五句是一个疑问句,表达了惊讶。第六句是鼓励性的建议,第七句则用"but"引导的并列句,展示了一个对比的情况。 通过这些练习题...

    哈工大c语言课件很多个

    哈工大c语言课件很多个

    27个经典趣味C++程序.rar

    27个经典趣味C++程序.rar

    谭浩强c语言word版

    谭浩强c语言word版

    c语言现代方法16章答案word版

    在 C 语言中,`typedef` 语句用于定义一个新的类型名。例如,`typedef struct { double real; double imaginary; } Complex;` 定义了一个名为 `Complex` 的类型,实际上是一个结构体,拥有两个成员变量 `real` 和 `...

    win10更改输入法切换方式(ctrl+空格)

    升级到win10后,很多更新还是特别喜欢的,但有一些更改有违以往的操作习惯,也到了让人发狂的地步,比如输入法切换,需要win+空格,特别不习惯,特别是对于it从业者更是如此,因为默认是中文输入法,大家都懂的 ...

    广东省惠州市惠东县2017_2018学年九年级英语全册unit11Sadmoviesmakemecry背诵卷新版人教新目标版20

    1. **使某人做某事**:这是一个动词短语,如"make me cry",表示某事或某人导致另一个人进行某个动作。 2. **宁愿做…而不愿做…**:表达个人倾向,例如"I would rather read books than watch TV.",意思是更喜欢...

    excel 中做的游戏“名字大战”

    相当有趣的名字大战,模拟迅雷等游戏制做,有击晕,发狂等攻击。奇乐无穷。

    c语言文件合集-很多文件

    c语言文件合集_很多文件

    AS3 MD5姓名大战 源码

    标题中的“AS3 MD5姓名大战”指的是一个使用ActionScript 3(简称AS3)编程语言开发的游戏,名为“姓名大战”。此游戏的核心玩法是通过随机匹配玩家的用户名或名称,将它们转化为游戏角色进行战斗。MD5算法在这里被...

    VS2010、VS2008等项目的默认浏览器修改方法(图文)

    那要如何修改调试时使用的默认浏览器呢?  默认情况下,VS会使用操作系统的默认浏览器,但我在调试 ASP.NET 程序...)这种方法真的不怎么样,有时可以有时不行,简直让人发狂。。。。。  这样就OK 以后就不用纠结了。

    gatsby-lotus-starter:这是我为了简化新Gatsby项目的创建而创建的入门程序

    盖茨比Lotus Starter 这是一个功能齐全的个人入门工具,我以此作为我的项目的基础,我认为这可能对我们社区中的其他人有用。 发狂,随时为改进做出贡献! [] 如果您有任何建议或错误要报告,请随时创建。特征盖茨比...

    Mybatis_day01.pdf

    MyBatis1. 目前最主流的持久层框架为hibernate与mybatis,而且国内目前情况使用Mybatis的公司比hibernate要多。 2. Hibernate学习门槛不低,要精通门槛更高...就算用hibernate的sqlquery,后续的维护工作也会让人发狂。

    浅谈do {…} while (0) 在宏定义中的作用

    下面就是一个例子: #define __set_task_state(tsk, state_value) \ do { (tsk)-&gt;state = (state_value); } while (0) 在Linux内核和其它一些著名的C库中有许多使用do{…}while(0)的宏定义。这种宏的用途是什么?...

    游戏策划文档范例.doc

    人类因过于依赖机器人,从而机器人的技术超速发展,人类试图创造完美的半生命体的机器人,但在其过程中,记忆系统被病毒感染,造成机器人发狂从实验室逃跑,拥有较高智商的机器人带领它的部下统治了半个大陆,建立了...

Global site tag (gtag.js) - Google Analytics