用 C 語言來玩 FP

C 和 FP 并不是對立的隧土!C 語言也可以實現(xiàn)一些比如類型推斷提针,泛型,函數(shù)閉包次洼,匿名函數(shù)等 FP 中的東西关贵。其實太陽底下其實無新鮮事,這次嘗試用 C 來玩 FP 吧卖毁!

GCC 為 C 添加了函數(shù)閉包的功能(在函數(shù)內定義函數(shù))揖曾,Clang 加入了 Block Type(可以實現(xiàn)類似匿名函數(shù)的功能)。

高階函數(shù)

一些基礎的高階函數(shù)例如 map亥啦、folderRight炭剪、folderLeft 等,用 C 語言也可以實現(xiàn)翔脱。

folder 函數(shù)

// f (arr[1],f(arr[2],f(arr[3],...f(arr[n],x0))))
// 例如 : 1+(2+(3+(4+x0)))
// f          => map 函數(shù)
// x0         => 初始值
// arr/length => 數(shù)組
int foldRight(int (*f)(int, int), int x0, int* const arr, int length) {
  int total  =x0;
  for (int i = length - 1; i >= 0; i--) {
    total = f(arr[i], total);
  }
  return total;
}

// foldLeft 同理
int foldLeft(int (*f)(int, int) ,int x0, int* const arr, int length) {
  int total = x0;
  for (int i = 0; i < length; i++) {
    total = f(total, arr[i]);
  }
  return total;
}

map 函數(shù)

// f => map 函數(shù)
void map(int (*f)(int), int* const arr ,int* const brr ,int length){
  for (int i = 1; i <= length; i++) brr[i] = f(arr[i]);
  return;
}

類似以上的寫函數(shù)的方法奴拦,還可以寫出另外一些高階函數(shù)比如filter,zip等届吁。

curring & closure

雖然柯里化現(xiàn)在有些專家對這個特性提出爭議错妖,但是還是要玩一下,畢竟柯里化幾乎 FP 的標配疚沐。

//add2(x)(y)=add(x,y)
int (*add2(int x))(int){
  int add1(int y){
    return add(x,y);
  }
  return add1;
}

//add3(x)(y)(z)=x+y+z
int (*(*add3(int x))(int))(int){
  int (*add2(int y))(int){
    int x1 = x;
    int add1(int z){
      return (x1 + y + z);
    }
    return add1;
  }
  return add2;
}

在函數(shù)內部定義函數(shù)這在標準C里面是做不到的暂氯,GCC里面實現(xiàn)函數(shù)的調用用的是彈床(trampoline)的方法,這個函數(shù)調用的方法是專為全局函數(shù)所設計的亮蛔。猜測是通過某種手段痴施,將局部函數(shù)提升為全局函數(shù)。

根據(jù)以上思路,可以寫一個函數(shù)將傳入的函數(shù)柯里化

//currying(f)(x)(y)=f(x,y)
int (*(*currying(int (*f)(int, int)))(int))(int){
  int (*fx(int x))(int){
    int (*f1)(int, int);
    f1 = f;
    int fxy(int y){
      return (f1(x, y));
    }
    return fxy;
  }
  return fx;
}

更大膽的嘗試

生硬的 Lambda

int main(int argc, const char **argv){
   int max, a, b;
   max = (scanf("%d %d", &a, &b), a > b ? a : b);
   printf("%d\n", max);
   return 0;
}

語句表達式可以結合宏使用辣吃,會寫出意想不到的功能动遭。

typeof()

感覺這里可能是 C 擴展借鑒了 C++11 里面的 decltype吧

int main(int argc, const char **argv){
  typeof(int (*)(int)) add(int x){
    int add(int y){return x + y;}
    return add;
  }  
  printf("%d\n", add(2)(3));
  return 0;
}

typeof()括號里還能放值,例如 typeof('a') c 相當于 char c

typeof神得、語句表達式和宏結合厘惦,還可以弄出類型推導。

Lambda

int main(int argc, const char **argv){
  int x = ({int trible(int x){return 3 * x;} trible;})(3);
  printf("%d\n", x);
  return 0;
}

對這一段代碼進行抽象循头,

({int trible(int x){return 3 * x;} trible;})

總結一下就是绵估,

({ 
   type name func_body
   name;
})

用宏實現(xiàn),

#define Lambda(type,body) ({\
  type lambda_funcname body\
  lambda_funcname; \
})

用上 typeof 卡骂,

#define Lambda(return_type,func_body) ({\
  typeof(return_type) lambda_funcname func_body\
  lambda_funcname; \
})

于是乎国裳,重寫 currying 函數(shù),

int (*(*currying(int (*f)(int, int)))(int))(int){
  return Lambda(int (*)(int),(int x){
    int (*f1)(int, int);
    f1 = f;
    return Lambda(int, (int y){
      return f1(x, y);
    });
  });
}

不過……代碼的可讀性并沒有提高全跨,反而更難讀了

總結

這種風格的 C 語言程序還是少寫為妙缝左,因為:

  1. 在 C 語言里面?zhèn)魅牒瘮?shù),傳出函數(shù)勢必使用很多的函數(shù)指針浓若,造成了代碼維護的困難渺杉;
  2. C 語言是一個靜態(tài)弱類型的語言,在類型系統(tǒng)上做得十分糟糕挪钓,容許類型的隱式轉化是越,也無法實現(xiàn)泛型與類型推導,這大大限制了這種風格代碼的應用廣度碌上,比如說前面的 map 函數(shù)倚评,不一定映射的是整數(shù),也可以是字符串馏予,但是因為只能聲明幾種天梧,大大限制了 map 的適用范圍;
  3. C 擴展對某些功能并不是原生支持的霞丧,可能藏著不少 bug呢岗。
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市蛹尝,隨后出現(xiàn)的幾起案子后豫,更是在濱河造成了極大的恐慌,老刑警劉巖突那,帶你破解...
    沈念sama閱讀 218,640評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件硬贯,死亡現(xiàn)場離奇詭異,居然都是意外死亡陨收,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,254評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來务漩,“玉大人拄衰,你說我怎么就攤上這事《牵” “怎么了翘悉?”我有些...
    開封第一講書人閱讀 165,011評論 0 355
  • 文/不壞的土叔 我叫張陵,是天一觀的道長居触。 經(jīng)常有香客問我妖混,道長,這世上最難降的妖魔是什么轮洋? 我笑而不...
    開封第一講書人閱讀 58,755評論 1 294
  • 正文 為了忘掉前任制市,我火速辦了婚禮,結果婚禮上弊予,老公的妹妹穿的比我還像新娘祥楣。我一直安慰自己,他們只是感情好汉柒,可當我...
    茶點故事閱讀 67,774評論 6 392
  • 文/花漫 我一把揭開白布误褪。 她就那樣靜靜地躺著,像睡著了一般碾褂。 火紅的嫁衣襯著肌膚如雪兽间。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,610評論 1 305
  • 那天正塌,我揣著相機與錄音嘀略,去河邊找鬼。 笑死传货,一個胖子當著我的面吹牛屎鳍,可吹牛的內容都是我干的。 我是一名探鬼主播问裕,決...
    沈念sama閱讀 40,352評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼逮壁,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了粮宛?” 一聲冷哼從身側響起窥淆,我...
    開封第一講書人閱讀 39,257評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎巍杈,沒想到半個月后忧饭,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,717評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡筷畦,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,894評論 3 336
  • 正文 我和宋清朗相戀三年词裤,在試婚紗的時候發(fā)現(xiàn)自己被綠了刺洒。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,021評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡吼砂,死狀恐怖逆航,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情渔肩,我是刑警寧澤因俐,帶...
    沈念sama閱讀 35,735評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站周偎,受9級特大地震影響抹剩,放射性物質發(fā)生泄漏。R本人自食惡果不足惜蓉坎,卻給世界環(huán)境...
    茶點故事閱讀 41,354評論 3 330
  • 文/蒙蒙 一澳眷、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧袍嬉,春花似錦境蔼、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,936評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至罐监,卻和暖如春吴藻,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背弓柱。 一陣腳步聲響...
    開封第一講書人閱讀 33,054評論 1 270
  • 我被黑心中介騙來泰國打工沟堡, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人矢空。 一個月前我還...
    沈念sama閱讀 48,224評論 3 371
  • 正文 我出身青樓航罗,卻偏偏與公主長得像,于是被迫代替她去往敵國和親屁药。 傳聞我的和親對象是個殘疾皇子粥血,可洞房花燭夜當晚...
    茶點故事閱讀 44,974評論 2 355

推薦閱讀更多精彩內容