【编程珠玑(续)】第一章 性能监视工具

一,计算素数

第一个程序:如果整数n,从2开始到n-1都不能整除则为素数。注意,n=2时一定是素数的判断

#include <iostream>
using namespace std;
int prime(int n)
{   
        int i;
        for (i = 2; i < n; i++)
                if(n%i == 0)
                        return 0;
        return 1;
}
int main()
{
    int i,n;
    n=1000;
    for(i = 2; i <= n; i++)
      if (prime(i))
          cout<<i<<" ";
}

第二个程序:只需要从2到sqrt(n)不能整除就可以

#include <iostream>
#include <cmath>
using namespace std;

int root(int n)  
{ 
    return (int) sqrt((float) n);  
} 
int prime(int n)
{   
        int i;
        for (i = 2; i < root(n); i++)  //每次循环都要调用 root()函数,可以申请一个变量,band = root(n)放到前面
                if(n%i == 0)
                        return 0;
        return 1;
}
int main()
{
    int i,n;
    n=1000;
    for(i = 2; i <= n; i++)
      if (prime(i))
          cout<<i<<" ";
} 


第三个程序:通过对被2、3和5整除的特殊检验,避免了近3/4的开方运算。语句计数表明,被2整除的性质大约把一半的输入归入合数,被3整除把剩余输入的1/3归入合数,被5整除再把剩下的这些数的1/5归入合数。其次,只考虑奇数作为可能的因子,在剩余的数中避免了大约一半的整除检验。它比程序P3大约快两倍,但是也比P3的错误更多。

#include <iostream>
#include <cmath>
using namespace std;

int root(int n)  
{ 
    return (int) sqrt((float) n);  
} 
int prime(int n)
{
        int i, bound;
        if (n % 2 == 0)
               return 0;
        if (n % 3 == 0)
               return 0;
        if (n % 5 == 0)
               return 0;
        bound = root(n);
        for (i = 7; i <= bound; i = i+2)
              if (n % i == 0)
                    return 0;
          return 1;
}
int main()
{
    int i,n;
    n=1000;
    for(i = 2; i <= n; i++)
      if (prime(i))
          cout<<i<<" ";
} 
以上程序,没有将素数:2,3,5检查出来

改进后程序如下:

#include <iostream>
#include <cmath>
using namespace std;

int root(int n)  
{ 
    return (int) sqrt((float) n);  
} 
int prime(int n)
{
        int i, bound;
        if (n % 2 == 0)
               return (n==2); //如果是2则返回 1 否则返回 0
        if (n % 3 == 0)
               return (n==3);
        if (n % 5 == 0)
               return (n==5);
        bound = root(n);
        for (i = 7; i <= bound; i = i+2)
              if (n % i == 0)
                    return 0;
          return 1;
}
int main()
{
    int i,n;
    n=1000;
    for(i = 2; i <= n; i++)
      if (prime(i))
          cout<<i<<" ";
} 

第四个程序:将费时的开方运算替换为 乘法,将第三个程序稍加调整

关键代码:

    

#include <iostream>
using namespace std;
 
int prime(int n)
{
        int i, bound;
        if (n % 2 == 0)
               return (n==2);
        if (n % 3 == 0)
               return (n==3);
        if (n % 5 == 0)
               return (n==5);

        for (i = 7; i*i <= n; i = i+2)
              if (n % i == 0)
                    return 0;
          return 1;
}
int main()
{
    int i,n;
    n=1000;
    for(i = 2; i <= n; i++)
      if (prime(i))
          cout<<i<<" ";
}


二,性能检测工具

        在linux系统下:

                 find  .  -name "*.c" |xargs wc -l
                 time — 执行命令并计时

                  time  find . -name "*.c" |xargs wc -l    // 统计执行命令时间

                  real          0m33.748s
                  user         0m0.772s
                  sys         0m1.044s

        1)实际时间(real time): 从command命令行开始执行到运行终止的消逝时间;

        2)用户CPU时间(user CPU time): 命令执行完成花费的用户CPU时间,即命令在用户态中执行时间总和;

        3)系统CPU时间(system CPU time): 命令执行完成花费的系统CPU时间,即命令在核心态中执行时间总和。

       其中,用户CPU时间和系统CPU时间之和为CPU时间,即命令占用CPU执行的时间总和。实际时间要大于CPU时间,因为Linux是多任务操作系统,往往在执行一条命令时,系统还要处理其它任务。

       另一个需要注意的问题是即使每次执行相同命令,但所花费的时间也是不一样,其花费时间是与系统运行相关的。


       代码的性能检测,统计每个函数调用次数。额可以检测哪个函数占用大量CPU时间,从而有效更改程序。


三,习题

       2)实现简单的埃氏筛法(Sieve of Eratosthenes)来计算所有小于n的素数。这个程序的主要数据结构是一个n比特的数组,初始值都为真。每发现一个素数时,数组中所有这个素数的倍数就设置为假。下一个素数就是数组中下一个为真的比特。

答案:下面的C程序实现了埃氏筛法来计算所有小于n的素数。其基本数据结构是n比特数组x,初始值全部为1。每发现一个素数,数组中所有它的倍数都设为0。下一个素数就是数组中的下一个取值为1的比特位。性能监视表明小于100000的素数有9592个,算法进行了大概2.57N次赋值。一般地,算法进行NloglogN次赋值;算法分析中涉及到答案1.1中的素数密度和调和数。下面是加上性能监视后的代码:

#include <iostream>
using namespace std;
 
int main()  
{    
    int i, p, n;  
    char x[100002];  
    n = 100000;  
    for (i = 1; i <= n; i++)  
             x[i] = 1;  
     x[1] = 0;  //1不是素数
     x[n+1] = 1;  
     p = 2;  
     while (p <= n) 
     {  
            cout<<p<<" "; 
            for (i = 2*p; i <= n; i = i+p)  
                  x[i] = 0;  
            do  
                 p++;  
             while (x[p] == 0);  //得到第一个标记为1的数,为素数
      }  
} 

    3) 一个简单的用于语句计数的性能监视工具每执行一个语句就增加一次计数。满足于更少的计数器可以同时减少监视程序的内存需求和运行时间。例如,可以给程序流图中的每个基本块关联一个计数器。也可以利用"基尔霍夫第一定律"进一步减少计数器的数量:如果你有一个if-then-else语句的计数器以及一个then分支语句的计数器,那么就不需要else分支语句的计数器了。


  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
### 回答1: 《编程珠玑 》是Peter Norvig在原作《编程珠玑》基础上的延,旨在进一步探讨编程技巧和优化方法。本书通过大量实例和案例,帮助读者更好地理解和运用编程的精髓。 本书主要涵盖了以下几个方面: 首先,作者介绍了一些高效的算法数据结构,以帮助读者更好地解决各类实际问题。例如,作者详细讲解了常用的排序算法和搜索算法,在实际应用中如何选择最合适的算法进行优化。 其次,本书还涉及了一些高级的编程技巧和思维模式。作者以实际案例为依据,深入讲解了如何进行代码重构、如何处理复杂的数据结构、如何进行并行计算等等。这些技巧和模式可以使读者的代码更加简洁、高效和可维护。 此外,本书还对一些热门的编程语言和框架进行了介绍和比较。作者通过对比分析,帮助读者选择最适合自己项目需求的编程语言和框架,并介绍了它们的一些优缺点和使用技巧。 最后,作者还分享了一些自己的编程心得和经验,并对未来的发展趋势进行了预测。他鼓励读者积极参与开源项目,不断学习和提升自己的编程水平。 总而言之,《编程珠玑 》是一本帮助读者深入理解编程精髓和优化技巧的实用指南。它丰富了原作的内容,并引入了新的案例和技巧,对于专业程序员和对编程感兴趣的人都是一本值得阅读的书籍。通过学习本书,读者能够更好地提升自己的编程能力,解决实际问题,并更好地适应行业的不断变化和发展。 ### 回答2: 《编程珠玑 》是一本继承《编程珠玑》精神的编程类图书,它深入探讨了更多关于编程算法的话题,帮助读者进一步提升编程技巧和解决问题的能力。 这本集书籍首先延了原版的思维方式和编程风格,鼓励读者通过实践和思考来掌握编程的本质。它从不同的角度和实际场景出发,提供了更多实用的编程技巧和解决问题的方法,使读者能够更加高效地编写代码。 《编程珠玑 》的内容涵盖了多个领域,包括排序算法、字符串处理、数据结构、网络编程等。它介绍了一些经典的算法数据结构,并通过大量的例子和实践题目帮助读者加深理解和掌握。 此外,《编程珠玑 》还关注了一些系统设计和性能优化的问题,提供了一些实际应用的案例和经验分享。通过学习这些内容,读者可以更好地设计和构建可扩展、高性能的软件系统。 总的来说,《编程珠玑 》是一本非常实用的编程类书籍,它以深入浅出的方式讲解了多个编程算法的关键概念,帮助读者在解决问题和编写代码时更具洞察力和技巧。无论是编程初学者还是有一定经验的开发工程师,都可以从中获得很多启发和收获。 ### 回答3: 《编程珠玑》是由Jon Bentley所著的计算机编程经典著作《编程珠玑》的篇。在这本作中,作者进一步探讨了计算机科学和编程的一些重要问题和技巧。 《编程珠玑》以问题为中心,通过讲解不同的编程问题和解决方案,培养读者的编程思维和解决问题的能力。书中的问题涉及各个领域,包括算法设计、数据结构性能优化、并发编程等内容,内容丰富而实用。 这本书的编写风格类似于《编程珠玑》,采用了一种琐碎而有趣的方式来讲解问题,引导读者逐步分析和优化解决方案。通过这种方法,读者可以深入了解各种编程技巧和策略,从而提高自己的编程水平。 《编程珠玑》还包含了许多实际案例和代码示例,读者可以通过实践来巩固所学的知识。这使得书籍的内容更加贴近实际编程应用,并能帮助读者更好地理解和运用所学的技巧。 总之,如果你对计算机科学和编程有浓厚的兴趣,并希望深入了解和掌握一些编程问题和技巧,那么《编程珠玑》是一本非常值得阅读的书籍。它将帮助你提升编程能力,并成为你在实际编程中的得力助手。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值