经典基础算法之面试题(系列一)(转)

1. 打靶问题的递归解法

Technorati 标记: 递归算法,字典法,二分法,打靶问题,日期问题,求幂,多项式求值

伦敦奥运会火热进行中,让我们来看个打靶的问题:一个射击运动员打靶,靶一共有10环,求连开10枪打中90环的可能行有多少种?
分析:这是一个典型递归求解问题。假设第10枪打x环,则将问题转换为剩下9枪打90-x环的可能有多少种,x的取值范围为[0, 10],根据加法原理,则:10枪打90环的可能 = 第10枪打0环,剩下9枪打90环的可能 + 第10枪打1环,剩下9枪打89环的可能 + 第10枪打2环,剩下9枪打88环的可能 
+ 第10枪打3环,剩下9枪打87环的可能 + 第10枪打4环,剩下9枪打86环的可能 + 第10枪打5环,剩下9枪打85环的可能 
+ 第10枪打6环,剩下9枪打84环的可能 + 第10枪打7环,剩下9枪打83环的可能 + 第10枪打8环,剩下9枪打82环的可能
+ 第10枪打9环,剩下9枪打81环的可能 + 第10枪打10环,剩下9枪打80环的可能。

                                                                         image

递归的停止条件为:
1. 若环数小于0 或者 剩下的环数大于剩下的枪数乘以10(即剩下每枪打10环用不玩所剩环数),则该递归路径不记入可能情况。
2. 若不满足条件1,且所剩枪数为1,则该递归路径记为1中可能情况。

 1:  #include <stdio.h>2:  3:  int recursion(int count, int score)4:  {5:      if (score < 0 || score > count * 10) {6:          return 0;7:      }8:   9:      if (1 ==  count) {
10:          return 1;
11:      }    
12:   
13:      int i = 0;
14:      int sum = 0;
15:   
16:      for (; i <= 10; ++i) {
17:          sum += recursion(count - 1, score - i);
18:      }
19:   
20:      return sum;
21:  }
22:   
23:  int main()
24:  {
25:      printf("result: %d\n", recursion(10, 90));        
26:      return 0;
27:  }

2. 求多项式的值与求幂的快速算法

2.1 求多项式的值

已知多项式中的系数a0,a1,a2……an及x的值,求f(x)。
image
多项式求值问题可以利用辗转相乘的方式进行计算,根据上面的式子分析,可以将上式转换为: f(x) = ((((an * x) + an-1) * x + an-2) * x  ……. + a1) * x + a0 
有了这个变形,代码自然而然就能写出来了:

 1:  #include <stdio.h>2:  3:  int polynomial(const int *coefficient, int n, int x)4:  {5:      int sum = 0;6:   7:      int i = n - 1;8:      for (; i != 0; --i) {9:          sum = sum * x + coefficient[i];
10:      }
11:   
12:      return sum * x + coefficient[0];
13:  }
14:   
15:  int main()
16:  {
17:      int coeff[5] = {1, 2, 3, 4, 5};
18:      printf("result: %d\n", polynomial(coeff, sizeof(coeff) / sizeof(coeff[0]), 10));
19:      return 0;
20:  }

2.2 求幂的快速算法

求x的p次幂本是简单的问题,可以将x次乘p次就可以了,这里我们当然不是要讨论这种计算方法,这里讨论的是怎样高效计算。
有没有可能减少做乘法的次数呢?让我们来做一个分析,考虑x的6次幂的情况:x * x * x * x * x * x = (x * x * x) * (x * x * x)
等式前面部分需进行6次乘法,后半部分需计算(x * x * x) ,然后乘以上次计算的值即可,共4次乘法。
通过上述分析可知,求x的p次幂可以通过递归折半的方法来减少乘法的次数,循着这个思考,实现就不太困难了。

 1:  #include <stdio.h>2:  3:  /* 统计乘法的次数 */4:  static int multiply_count = 0;5:   6:  /* 求x的p次幂 */7:  int power(int x, int p)8:  {9:      int ret =  0;
10:   
11:      if (1 == p) {
12:          return x;
13:      }
14:   
15:      /* 先计算x的p/2次幂 */
16:  ret = power(x, p >> 1);
17:      ret *= ret;
18:      ++multiply_count;
19:   
20:      /* 若p是奇数,则再乘一次x */
21:  if (0 != p % 2 /* 求模本身是一个耗性能的运算,这里可优化为 p – ((p >> 1) << 1) */) { 
22:          ret *= x;
23:          ++multiply_count;
24:      }    
25:   
26:      return ret;
27:  }
28:   
29:  int main()
30:  {
31:      printf("resulut: %d\n", power(2, 20));    
32:      printf("multiply count: %d\n", multiply_count);
33:      return 0;
34:  }

计算2的20次幂总共进行5次乘法就足够了,而原始的算法需要20次,那么这种算法究竟需要多少次乘法呢?
由于该算法根本思想是折半递归,类似于2分查找,所以乘法的次数为lgp取天花板值这个数量级的(lgp表示以2为底p的对数)。
整型int最多表示2的32次幂,因此最多节省24次乘法,对于现代的计算机,这似乎不是特别重要,但是在以下情况下该算法具有重要价值:
1. 对于需要高频率计算幂的情况;
2. 对于大数高精度计算的情况,如需计算2的10000次幂。
当然了,这也是对递归算法和2分法的巧妙应用,学习其思想吧。

3. 一年中的第n天是几月几号?

和这个问题类似的问题还有:
1. 给定某年某月某日,问这是这年的第多少天?
2. 已知某年的1月1号是星期几,求给定的某年某月某日是星期几?

其实这类问题的共同点在于它们都需要考虑闰年问题,大月小月问题;
首先我们解决闰年问题,根据闰年的定义定义如下宏来判断某年是否是闰年:
#define IS_LEAP(X) (((X) % 400 == 0 || (X) % 100 != 0 && (X) % 4 == 0) ? 1 : 0)

在来看大月小月问题,可以定义如下二维数组,用day_count_of_month[0][12]表示润年情况下每月的天数,用day_count_of_month[1][12]表示非润年情况下每月的天数(这就是传说中的字典法了,就是根据提供的信息查表,类似于查字典,所以叫做字典法):
int day_count_of_month[2][12] = {

                                                {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}, 
                                                {31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}

                                               };

1:  #include <stdio.h> 2:  3:  #define IS_LEAP(X) (((X) % 400 == 0 || (X) % 100 != 0 && (X) % 4 == 0) ? 1 : 0)4:  int day_count_of_month[2][12] = {{31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}, 5:                                   {31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31} }; 6:  /* 计算year年的第n天是几月几号 */7:  void convert_to_date(int year, int n) 8:  { 9:      int leap = IS_LEAP(year); 
10:      if(year < 1970 || year > 5000 || n <= 0 || n > 365 + leap) { 
11:          printf("bad year\n"); 
12:      }
13:   
14:      int i = 0; 
15:      for(; n > day_count_of_month[leap][i]; ++i) { 
16:          n -= day_count_of_month[leap][i]; 
17:      } 
18:   
19:      printf("year : %d, month : %d, day : %d\n", year, i + 1, n); 
20:  } 
21:   
22:  int main() 
23:  { 
24:      convert_to_date(2012, 230); 
25:  } 

上面这段代码解决了第一个问题,第二个问题可以转换为类似第一个问题的问题:
先计算给定的某年某月某日到给定已知这天的总天数s=>s%7=>利用模运算结果来推算所求日期的星期情况即可。

 

http://www.cnblogs.com/dskit/archive/2012/08/11/2633736.html

 

转载于:https://www.cnblogs.com/bizhu/archive/2012/08/19/2646325.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/461769.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2013年11月份我国网络不良与垃圾短信息分析报告

IDC评述网&#xff08;idcps.com&#xff09;12月17日报道&#xff1a;近日&#xff0c;12321网络不良与垃圾信息举报受理中心最新播报了11月份我国不良与垃圾短信息发送情况。根据报告显示&#xff0c;11月份垃圾邮件发送IP地址与上月大体相同&#xff0c;主要集中在湖北、北京…

字节对齐的写法.

What I write, what I lose. 之前看到Nginx的code的时候. 看到一个关于字节对齐的实现. 突然想起之前去面试的时候, 别人让我实现一个4字节对齐. 当然奇怪别人怎么问这种问题. 这时候开始有一点点体会. 以下将个人总结的几种写法写上. #define common_align1(x) ((x/4((x%4)?…

SCOM Rule 介绍 [SCOM中文系列之六]

SCOM的Rule主要用来收集数据&#xff0c;我们看到的EventView&#xff0c;PerformanceData&#xff0c;Report所需数据大部分都是通过Rule来收集的。 SCOM Rule的基础结构是触发指定的条件&#xff0c;就执行指定的动作。这个我们通过Rule的属性界面可以看的清楚点&#xff1a;…

init进程的详解

以下内容源于朱有鹏嵌入式课程的学习与整理&#xff0c;如有侵权请告知删除。 参考博客 内核源码——C语言阶段的start_kernel函数_天糊土的博客-CSDN博客 分析根文件系统中的/linuxrc文件_天糊土的博客-CSDN博客 linux内核sys_mount()分析_kai_ding的博客-CSDN博客_sys_mount…

跟着石头哥哥学cocos2d-x(四)--cocos2dx中的动画以及TexturePacker使用

2019独角兽企业重金招聘Python工程师标准>>> 之前向Andreas Loew申请了一枚TexturePacker注册码&#xff0c;很快都下来了&#xff0c;作为回报我打算还是写一篇关于TexturePacker的使用博客吧&#xff0c;有兴趣的可以在这里申请密钥,http://www.codeandweb.com&am…

【C/C++学习】之七、指向函数的指针

什么是指向函数的指针 函数指针是指向函数的指针变量&#xff0c;不是指向对象的指针&#xff01;函数指针本身应该是“指针变量”&#xff1b; “在C语言中&#xff0c;函数本身不是变量&#xff0c;但可以定义指向函数的指针&#xff0c;这种指针可以被赋值、存放于数组之中&…

头像和Karma汽车

Powered by Zoundry RavenTechnorati : Karma, 头像, 汽车 Del.icio.us : Karma, 头像, 汽车 转载于:https://www.cnblogs.com/bsmagic/archive/2008/08/06/1261858.html

内核中架构相关代码简介

以下内容源于朱友鹏《物联网大讲堂》课程的学习&#xff0c;如有侵权&#xff0c;请告知删除。 1、内核代码基本分为3块 &#xff08;1&#xff09;arch&#xff1a;本目录下全是cpu架构有关的代码 &#xff08;2&#xff09;drivers&#xff1a;本目录下全是硬件的驱动 &am…

All-In-One Code Framework [一站式示例代码库] 【转】

All-In-One Code Framework [一站式示例代码库]2010 对一站式示例代码库&#xff0c;对奋战在一站式示例代码库上的每一位工程师来说都是不同寻常的一年。 在我们共同努力和开发社区的支持下&#xff0c;该项目从一年前的草根雏形&#xff0c;成长为CodePlex排名前三&#xff0…

Repeater的嵌套

今天做了一个小项目学习一下&#xff0c;是关于两个Repeater的嵌套使用的&#xff1a; 关于Repeater的嵌套&#xff0c;关键在于数据项的绑定。 首先&#xff0c;外Repeater需要设置OnItemDataBound事件&#xff0c;在事件中添加子Repeater的数据项绑定&#xff1b; <asp:Re…

Ioc容器Autofac介绍

Autofac是轻量级的开源Ioc容器&#xff0c;在这里可以下载http://code.google.com/p/autofac/。如果你用过其他的Ioc容器&#xff0c;那么学习Autofac使用也会比较容易&#xff0c;下面将通过一些例子来讲解其用法。 先看一个例子&#xff1a; 首先新建一个工程&#xff0c;添加…

汇编指令的学习2——常用的ARM指令

一、常用ARM指令1&#xff1a;数据处理指令 &#xff08;1&#xff09;数据传输指令 mov mvn&#xff08;源目标按位取反后赋给目标&#xff09; &#xff08;2&#xff09;算术指令 add sub rsb adc sbc rsc &#xff08;3&#xff09;逻辑指令 and orr eor …

cocos2dx 3.4 截图代码

Size size Director::sharedDirector()->getWinSize(); //定义一个屏幕大小的渲染纹理 RenderTexture* pScreen RenderTexture::create(size.width,size.height, kCCTexture2DPixelFormat_RGBA8888); log("-------log1---");//获得当前的场景指针 Scene* pC…

NSURLRequest详解IOS最基础的api

http://blog.csdn.net/bl1988530/article/details/6590099转载于:https://www.cnblogs.com/qiqibo/archive/2012/08/22/2650996.html

正则表达式 Mather类的使用

Matcher类: 使用Matcher类,最重要的一个概念必须清楚:组(Group),在正则表达式中 ()定义了一个组,由于一个正则表达式可以包含很多的组,所以下面先说说怎么划分组的, 以及这些组和组的下标怎么对应的. 下面我们看看一个小例子,来说明这个问题 \w(\d\d)(\w) 这个正则表达式有三…

ARM协处理器的汇编指令

以下内容源于网络资源的学习与整理&#xff0c;如有侵权请告知删除。 一、协处理器的简介 协处理器&#xff08;cp&#xff0c;coprocessor&#xff09;是SoC内部的处理单元&#xff0c;用来协助主CPU实现一些特定的功能&#xff0c;比如MMU、cache、TLB等内容。ARM在设计上支…

用代码实现Sharepoint2010的个人信息的照片上传(2)(原创)

用代码从AD读取照片信息并同步到SharePoint2010的个人照片信息中 前言 上篇我们讲到用代码实现了照片信息同步到SharePoint2010的个人照片信息中&#xff0c;但是做的是从文件夹读取照片信息到SharePoint2010,今天我们要做的从AD的thumbnailPhoto属性中读取照片信息到SharePoin…

HTML5实现Word中文字全环绕图片效果

实现这个功能是当时看见数字报的展示出来的编排方式&#xff0c;就是图片热点&#xff0c;而且下载打开还是PDF的&#xff0c;让我这个WP用户咋个看。想实现Word中的编排方式&#xff0c;也需要浏览器支持Word插件&#xff0c;火狐、谷歌、Opera这些都要另外弄插件&#xff0c;…

汇编指令的学习4——ldm/stm指令、栈的处理

1、为什么需要多寄存器访问指令&#xff1f; ldr/str每周期只能访问4字节内存&#xff0c;如果需要批量读取、写入内存时太慢&#xff0c;解决方案是stm/ldm ldm (load register mutiple) stm&#xff08;store register mutiple&#xff09; 2、举例&#xff1a;stmia sp,…

第一课 GCC入门

1序言 gcc是一个可移植的编译器&#xff0c;支持多种硬件平台&#xff1b;也不仅仅是一个本地编译器也是一个跨平台编译器&#xff1b;支持多张语言编译时按照模块化设计支持多种语言。 gcc编译过程&#xff1a;预处理&#xff08;预处理器&#xff09;&#xff1b;编译&#x…