• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            posts - 297,  comments - 15,  trackbacks - 0

            http://zhedahht.blog.163.com/blog/static/2541117420072915131422/題目:求1+2+…+n,要求不能使用乘除法、forwhile、if、elseswitchcase等關鍵字以及條件判斷語句(A?B:C)。

            分析:這道題沒有多少實際意義,因為在軟件開發中不會有這么變態的限制。但這道題卻能有效地考查發散思維能力,而發散思維能力能反映出對編程相關技術理解的深刻程度。

            通常求1+2+…+n除了用公式n(n+1)/2之外,無外乎循環和遞歸兩種思路。由于已經明確限制forwhile的使用,循環已經不能再用了。同樣,遞歸函數也需要用if語句或者條件判斷語句來判斷是繼續遞歸下去還是終止遞歸,但現在題目已經不允許使用這兩種語句了。

            我們仍然圍繞循環做文章。循環只是讓相同的代碼執行n遍而已,我們完全可以不用forwhile達到這個效果。比如定義一個類,我們new一含有n個這種類型元素的數組,那么該類的構造函數將確定會被調用n次。我們可以將需要執行的代碼放到構造函數里。如下代碼正是基于這個思路:

            class Temp
            {
            public:
                  Temp() { ++ N; Sum += N; }

                  static void Reset() { N = 0; Sum = 0; }
                  static int GetSum() { return Sum; }

            private:
                  static int N;
                  static int Sum;
            };

            int Temp::N = 0;
            int Temp::Sum = 0;

            int solution1_Sum(int n)
            {
                  Temp::Reset();

                  Temp *a = new Temp[n];
                  delete []a;
                  a = 0;

                  return Temp::GetSum();
            }

            我們同樣也可以圍繞遞歸做文章。既然不能判斷是不是應該終止遞歸,我們不妨定義兩個函數。一個函數充當遞歸函數的角色,另一個函數處理終止遞歸的情況,我們需要做的就是在兩個函數里二選一。從二選一我們很自然的想到布爾變量,比如ture1)的時候調用第一個函數,false0)的時候調用第二個函數。那現在的問題是如和把數值變量n轉換成布爾值。如果對n連續做兩次反運算,即!!n,那么非零的n轉換為true,0轉換為false。有了上述分析,我們再來看下面的代碼:

            class A;
            A* Array[2];

            class A
            {
            public:
                  virtual int Sum (int n) { return 0; }
            };

            class B: public A
            {
            public:
                  virtual int Sum (int n) { return Array[!!n]->Sum(n-1)+n; }
            };

            int solution2_Sum(int n)
            {
                  A a;
                  B b;
                  Array[0] = &a;
                  Array[1] = &b;

                  int value = Array[1]->Sum(n);

                  return value;
            }

            這種方法是用虛函數來實現函數的選擇。當n不為零時,執行函數B::Sum;當n0時,執行A::Sum。我們也可以直接用函數指針數組,這樣可能還更直接一些:

            typedef int (*fun)(int);

            int solution3_f1(int i) 
            {
                  return 0;
            }

            int solution3_f2(int i)
            {
                  fun f[2]={solution3_f1, solution3_f2}; 
                  return i+f[!!i](i-1);
            }

            另外我們還可以讓編譯器幫我們來完成類似于遞歸的運算,比如如下代碼:

            template <int n> struct solution4_Sum
            {
                  enum Value { N = solution4_Sum<n - 1>::N + n};
            };

            template <> struct solution4_Sum<1>
            {
                  enum Value { N = 1};
            };

            solution4_Sum<100>::N就是1+2+...+100的結果。當編譯器看到solution4_Sum<100>時,就是為模板類solution4_Sum以參數100生成該類型的代碼。但以100為參數的類型需要得到以99為參數的類型,因為solution4_Sum<100>::N=solution4_Sum<99>::N+100。這個過程會遞歸一直到參數為1的類型,由于該類型已經顯式定義,編譯器無需生成,遞歸編譯到此結束。由于這個過程是在編譯過程中完成的,因此要求輸入n必須是在編譯期間就能確定,不能動態輸入。這是該方法最大的缺點。而且編譯器對遞歸編譯代碼的遞歸深度是有限制的,也就是要求n不能太大。

            大家還有更多、更巧妙的思路嗎?歡迎討論^_^

            本文已經收錄到《劍指Offer——名企面試官精講典型編程題》一書中,有改動。歡迎關注。我把這篇博客翻譯成了英文,感興趣的朋友可以到

            http://codercareer.blogspot.com/2011/10/no-08-calculate-12n.html查看。

            博主何海濤對本博客文章享有版權。網絡轉載請注明出處http://zhedahht.blog.163.com/。整理出版物請和作者聯系。對解題思路有任何建議,歡迎在評論中告知,或者加我微博http://weibo.com/zhedahht或者http://t.163.com/zhedahht與我討論。謝謝。
            from:
            http://zhedahht.blog.163.com/blog/static/2541117420072915131422/

            posted on 2012-07-05 10:04 chatler 閱讀(471) 評論(0)  編輯 收藏 引用 所屬分類: C++_BASIS
            <2025年8月>
            272829303112
            3456789
            10111213141516
            17181920212223
            24252627282930
            31123456

            常用鏈接

            留言簿(10)

            隨筆分類(307)

            隨筆檔案(297)

            algorithm

            Books_Free_Online

            C++

            database

            Linux

            Linux shell

            linux socket

            misce

            • cloudward
            • 感覺這個博客還是不錯,雖然做的東西和我不大相關,覺得看看還是有好處的

            network

            OSS

            • Google Android
            • Android is a software stack for mobile devices that includes an operating system, middleware and key applications. This early look at the Android SDK provides the tools and APIs necessary to begin developing applications on the Android platform using the Java programming language.
            • os161 file list

            overall

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            久久国产精品国产自线拍免费| 99久久精品国产一区二区三区| 精品国产乱码久久久久软件| 无码人妻久久一区二区三区免费 | 亚洲?V乱码久久精品蜜桃 | 久久综合欧美成人| 人妻无码精品久久亚瑟影视| 久久综合给合久久国产免费| 国产农村妇女毛片精品久久| 久久精品国产99国产精品导航 | 久久露脸国产精品| 久久精品国产亚洲av日韩| 久久精品中文字幕有码| 久久久久99精品成人片直播| 老司机午夜网站国内精品久久久久久久久 | 久久这里只有精品18| 欧美无乱码久久久免费午夜一区二区三区中文字幕 | 亚洲中文久久精品无码ww16| 人人狠狠综合久久亚洲88| 久久无码AV一区二区三区| 国产激情久久久久影院老熟女免费| 无码日韩人妻精品久久蜜桃| 久久久久久精品成人免费图片| 久久久久亚洲AV无码去区首| 99久久免费国产精精品| 久久亚洲中文字幕精品有坂深雪| 久久国产欧美日韩精品| 人人狠狠综合88综合久久| 精品久久人人妻人人做精品| 久久国产精品成人影院| 久久综合国产乱子伦精品免费| 伊人久久大香线蕉av不卡| 久久国产欧美日韩精品| 日产精品99久久久久久| 久久久久人妻一区精品性色av | 久久综合丁香激情久久| 97久久精品午夜一区二区| 久久天天躁狠狠躁夜夜网站 | 亚洲伊人久久大香线蕉苏妲己| 91精品国产9l久久久久| 精品人妻伦九区久久AAA片69 |