中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

C++模版編程實現Haskell的函數模式匹配特性[圖]

發布時間:2020-08-06 14:38:50 來源:網絡 閱讀:504 作者:編程小博客 欄目:編程語言

C++模版編程實現Haskell的函數模式匹配特性[圖]:
大神 Bartosz Milewski 在2009年寫了一篇文章《What Does Haskell Have to Do with C++?》,使用C++實現Haskell函數式編程語言的一些特性。【傳送門在文末】
其中有這樣一段例子:
// code 1
1.template<int n>class fact {
2.public:

  1. staticconstint value = n * fact<n -1>::value;
    4.};
  2. 6.template<>class fact<0>{// specialization for n = 0
    7.public:

  3. staticconstint value =1;
    9.};
    注:原文中使用的是struct關鍵字,這里改為class并加上了public
    我猜,你沒看懂。沒關系,我們先跳過上面這一段有著【令人恐怖的語法】的C++模版代碼。
    上面的例子想干嘛呢?其實它只是想計算n的階乘。
    C++模版編程實現Haskell的函數模式匹配特性[圖]
    如果你在C語言里面學過遞歸,應該知道下面這段計算階乘的遞歸函數
    // code 2
    int fact(int n){
    if(0== n )
    return1; //0階問題答案。0! 等于1
    else
    return( n fact( n -1)); //問題降階:n階->n-1階
    }
    它的效果就等于下面的代碼
    // code 3
    int fact2(int n){ // 用 for 循環計算階乘
    int p =1;
    for(int i=n; i >=1; i--)
    p
    = i;
    return p;
    }
    那么,第一段代碼(code1)與第二段代碼(code2)的區別在哪里呢?
    區別在于,code1是在編譯時(由編譯器)計算的,code2是在運行時(就是代碼運行的時候)計算的。
    現在來解釋一下code1 (部分根據Bartosz Milewski文中的說法)
    // code 1
    / 第1行代碼聲明了一個類模版 fact。
    這個模版接受一個“非類型參數”n,
    n是整數。
    /
    1.template<int n>class fact {
    2.public:
    / 第3行代碼聲明了一個靜態整型常量
    成員 value。而 value 的值是使用
    遞歸模版表示的
    /
  4. staticconstint value = n * fact<n -1>::value;
    4.};
  5. / 第6行代碼是“特化”類模版fact,
    也就是顯式地給出某種類型參數的
    類模板的一個實例的代碼,而非由
    編譯器生成。
    在這里,是給出了參數n為0時模板
    fact的代碼。這樣,編譯器不會再
    根據類模版fact生成n=0時的代碼
    關于模版特化,詳見文末鏈接
    /
    6.template<>class fact<0>{// specialization for n = 0
    7.public:

  6. static const int value = 1;
  7. };
    / 根據C++規范,模版特化的代碼必須
    放到模版聲明之后。
    因此上面的代碼看上去好像先處理了
    由n階到n-1階的降階問題,然后再給
    出了0階的解答
    這可不像code2。code2中有if/else,
    因此可以把降階代碼與0階解答代碼調
    換先后次序(當然if條件得改)。
    /
    那么這個用模版計算階乘的代碼(類?)該怎么用呢?如下:
    cout <<"Factorial of 0 = "<< fact<0>::value << endl;
    其中,C++編譯器會為“fact<0>::value”這個調用匹配最合適的模版代碼,也就是code1中的第6-9行代碼。
    如果用非零參數調用呢?
    cout <<"Factorial of 8 = "<< fact<8>::value << endl;
    其中,C++編譯器會為“fact<8>::value”這個調用匹配code1中的第1-4行代碼。
    前面blahblhaaaaaaaaaaaah講了一大堆,其實都不是正經事兒。
    正經是下面的Haskell代碼:
    //code 4
  8. fact 0=1
  9. fact n = n * fact (n -1)
    上面兩行代碼定義了函數fact。fact是函數名,fact的后面、等號的前面是函數的參數。等號后面是函數體,函數體的計算結果就是fact函數的返回值。
    當程序員調用【fact 8】的時候(參數是8,因為Haskell函數調用一般不像C++那樣給參數加括號),Haskell會將之匹配到上面代碼的第2行。誰動了我的奶酪讀書筆記(http://www.simayi.net/dushubiji/6208.html)摘抄好詞好句及感悟賞析,這種參數匹配,是Haskell特有的函數聲明與調用方式。

    所以前面的code1中C++模版代碼,就是在模仿 code4 中的Haskell代碼。
    下面給出一個完整的Haskell程序
    moduleFactwhere
    importSystem.IO
    fact::Integer->Integer
    fact0=1
    fact n = n * fact (n-1)
    main::IO()
    main=do
    putStrLn $"8! = "++ show (fact 8)
    putStrLn $"88! = "++ show (fact 88)
    上面的代碼輸出結果是:
    8! = 40320
    88! =185482642257398439114796845645546284380220968949399346684421580986889562184028199319100141244804501828416633516851200000000000000000000
    Haskell對C++說:我能算88!,你行嗎?
    C++說:你欺負人!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

伊宁市| 湘阴县| 繁昌县| 金堂县| 玛曲县| 长顺县| 通化县| 米易县| 阳城县| 遂溪县| 铜山县| 府谷县| 泰顺县| 喀什市| 瓦房店市| 上林县| 阿克陶县| 闸北区| 桐乡市| 盐津县| 吉木乃县| 塘沽区| 台北县| 北京市| 丰宁| 合山市| 万宁市| 东平县| 福州市| 循化| 宁波市| 汪清县| 怀安县| 类乌齐县| 马尔康县| 彭泽县| 平罗县| 龙口市| 长子县| 南阳市| 康马县|