浅谈《剑指offer》原题:不使用条件、循环语句求1+2+……+n
如侵犯您的版权,请联系:windeal12@qq.com
《剑指offer》上的一道原题,求1+2+……+n,要求不能使用乘除法,for、while、if、else、switch、case等关键字以及条件判断语句(a?b:c)。
第一次看到这道题大约有一年的时间了,在霸笔网易的时候,当时我就晕了。。。心想这是神马东西,后来发现这是原题!!然后后悔自己没看过书了。。。
《剑指offer》上给出了不错的解法,但是这里有个解法更巧妙,虽然技术含量不高,但是可以参考,这就是《程序员面试笔试宝典》中所给出的答案。
解法一:利用宏定义求解
假设n=1000。相信看到答案的你们都会笑了。
- #include <stdio.h>
- #define L sum+=(++n);
- #define K L;L;L;L;L;L;L;L;L;L;
- #define J K;K;K;K;K;K;K;K;K;K;
- #define H J;J;J;J;J;J;J;J;J;J;
- int main()
- {
- int sum = 0;
- int n = 0;
- H;
- printf("%d\n", sum);
- return 0;
- }
怎么样!有木有很搞笑。。。。
解法二:利用构造函数
实际上就是利用类里面的静态成员变量,然后通过构造函数去调用。其实对于c++掌握熟练的人来说,也可以很轻松的明白。
- #include <stdio.h>
- #include <iostream>
- using namespace std;
- class Temp
- {
- public:
- Temp(){++N; Sum += N;}
- static void Reset(){N=0; Sum=0;}
- static unsigned int GetSum(){return Sum;}
- private:
- static unsigned int N;
- static unsigned int Sum;
- };
- unsigned int Temp::N = 0;
- unsigned int Temp::Sum = 0;
- unsigned int Sum_Solution1(unsigned int n)
- {
- Temp::Reset();
- Temp *a = new Temp[n];
- delete []a;
- a = NULL;
- return Temp::GetSum();
- }
- int main()
- {
- printf("%d\n", Sum_Solution1(1000));
- return 0;
- }
解法三:利用虚函数求解
这也利用了多态的性质,特别巧妙。
- #include <stdio.h>
- #include <iostream>
- using namespace std;
- class A;
- A* Array[2];
- class A
- {
- public:
- virtual unsigned int Sum(unsigned int n)
- {
- return 0;
- }
- };
- class B:public A
- {
- public:
- virtual unsigned int Sum(unsigned int n)
- {
- return Array[!!n]->Sum(n-1) + n;
- }
- };
- int Sum_Solutiion2(int n)
- {
- A a;
- B b;
- Array[0] = &a;
- Array[1] = &b;
- int value = Array[1]->Sum(n);
- return value;
- }
- int main()
- {
- printf("%d\n", Sum_Solutiion2(1000));
- return 0;
- }
这种思路基于虚函数来实现函数的选择,当n不为0的时候,一直调用的是B::Sum();当n等于0时,调用的就是函数A::Sum()。
解法四:利用函数指针求解
在纯C语言的编程环境中,我们不能使用虚函数,这时候函数指针就可以达到一样的效果了!
- #include <stdio.h>
- typedef unsigned int (*fun)(unsigned int);
- unsigned int Sum_Solutiion3_Teminator(unsigned int n)
- {
- return 0;
- }
- unsigned int Sum_Solutiion3(unsigned int n)
- {
- static fun f[2] = {Sum_Solutiion3_Teminator, Sum_Solutiion3};
- return n + f[!!n](n-1);
- }
- int main()
- {
- printf("%d\n", Sum_Solutiion3(1000));
- return 0;
- }
解法五:利用模板类型来求解
本质都是多态。可惜不是所有编译器都支持,VC++6.0就不支持。。
- #include <stdio.h>
- #include <iostream>
- template <unsigned int n>struct Sum_Solutiion4
- {
- enum Value{N = Sum_Solutiion4<N-1>::N + n};
- };
- template <> struct Sum_Solutiion4<1>
- {
- enum Value{N = 1};
- };
- int main()
- {
- printf("%d\n", Sum_Solutiion4<1000>::N);
- return 0;
- }
感觉这道题这些方法都能理解的话,说明c++水平已经不错了,我当时第一次看见都是云里雾里的!
浅谈《剑指offer》原题:不使用条件、循环语句求1+2+……+n的更多相关文章
- 剑指 offer 第一题: 二维数组中的查找
打算写 图解剑指 offer 66 题 的系列文章,不知道大家有没有兴趣
- 剑指Offer编程题2——替换空格
剑指Offer编程题2——替换空格 题目描述 请实现一个函数,将一个字符串中的每个空格替换成“%20”.例如,当字符串为We Are Happy.则经过替换之后的字符串为We%20Are%20Happ ...
- 剑指Offer编程题1——二维数组中的查找
剑指Offer编程题1---------------二维数组中的查找 题目描述 在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完 ...
- 剑指offer编程题66道题 36-66
36.两个链表的第一个公共节点 题目描述 输入两个链表,找出它们的第一个公共结点. 1.具有重合节点的两个链表是一个Y字性,用两个堆栈放这两个链表,从尾部开始遍历,直到遍历到最后一个重合节点. 这种算 ...
- 牛客网剑指offer刷题总结
二维数组中的查找: 题目描述:在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数. 两 ...
- LeetCode剑指Offer刷题总结(一)
LeetCode过程中值得反思的细节 以下题号均指LeetCode剑指offer题库中的题号 本文章将每周定期更新,当内容达到10题左右时将会开下一节. 二维数组越界问题04 public stati ...
- 剑指offer刷题
1.面试题43. 1-n整数中1出现的次数 输入一个整数 n ,求1-n这n个整数的十进制表示中1出现的次数. 例如,输入12,1-12这些整数中包含1 的数字有1.10.11和12,1一共出现了5次 ...
- 剑指offer编程题Java实现——面试题12相关题大数的加法、减法、乘法问题的实现
用字符串或者数组表示大数是一种很简单有效的表示方式.在打印1到最大的n为数的问题上采用的是使用数组表示大数的方式.在相关题实现任意两个整数的加法.减法.乘法的实现中,采用字符串对大数进行表示,不过在具 ...
- 剑指offer刷题(Tree)
开篇 二刷剑指offer了,本来用Tyora记的笔记,发现字数到四万了就变得好卡o(╥﹏╥)o,刚好开始写博客,就转过来吧,记下来子自己看.不废话,开刷... JZ26. 树的子结构 输入两棵二叉树A ...
- 剑指offer编程题Java实现——替换空格
题目描述 请实现一个函数,将一个字符串中的空格替换成"%20".例如,当字符串为We Are Happy.则经过替换之后的字符串为We%20Are%20Happy. package ...
随机推荐
- [原创] css中的绝对定位和相对定位
我对博客的认识是:记录问题,解决问题,分享知识.如果有轮子,我不需要造轮子. 首先,定位无论是相对定位还是绝对定位,必须有一个参考项,而这个参考项,专业术语称之为 包含块,这里的包含块是指在定位时 ...
- 高通camera结构(摄像头基础介绍)
摄像头基础介绍 一.摄像头结构和工作原理. 拍摄景物通过镜头,将生成的光学图像投射到传感器上,然后光学图像被转换成电信号,电信号再经过模数转换变为数字信号,数字信号经过DSP加工处理,再被送到电脑中进 ...
- MYSQL数据库里面的所有密码批量MD5加密
如果你的字段够长度的话:UPDATE users SET password = MD5(password);如果长度不够,可以先增加长度后再做,或者多建一列,完成后删除原来的列!(如passwd)UP ...
- 深度学习笔记(一):logistic分类【转】
本文转载自:https://blog.csdn.net/u014595019/article/details/52554582 这个系列主要记录我在学习各个深度学习算法时候的笔记,因为之前已经学过大概 ...
- 快速升级openwrt的linux内核版本
一.分析 要升级openwrt的linux内核版本,关键是要制作内核配置文件 二.内核配置文件制作方法 2.1当前openwrt对应的某个开发板有对应的内核配置文件,比如此时的openwrt的linu ...
- jquery map方法
jQuery.map( array, callback(elementOfArray, indexInArray) )Returns: Array 感觉jquery的map方法非常好用,特向大家分享下 ...
- linux下错误 && 解决方法
1.使用yum命令安装出现错误 Error: Cannot find a valid baseurl for repo: extras 解决方法: vi /etc/resolv.conf 在此文件最后 ...
- jq定时器
1.$(function(){ setInterval (showTime, 2000); function showTime(){ var today = new Date(); alert(&qu ...
- RabbitMQ入门(2)——工作队列
前面介绍了队列接收和发送消息,这篇将学习如何创建一个工作队列来处理在多个消费者之间分配耗时的任务.工作队列(work queue),又称任务队列(task queue). 工作队列的目的是为了避免立刻 ...
- 技术分享:JS工作原理
一 浏览器组成可分两部分:Shell+内核. 浏览器内核又可以分成两部分:渲染引擎(layout engineer或者Rendering Engine)和JS引擎. 渲染引擎功能作用 渲染引擎,负责对 ...