LCTHW - 18

18. 函數(shù)指針

C語言中的函數(shù)實際上都是指向一段代碼的指針(這段代碼就是函數(shù)的實現(xiàn)).就像創(chuàng)建指向結構體, 變量的指針一樣, 你也可以把一個指針指向函數(shù). 函數(shù)指針的主要用途在于給其他函數(shù)傳"回調(diào)"(callback), 或者用來模擬類和對象.

函數(shù)指針的形式是這樣的:

int (*POINTER_NAME)(int a, int b)

和函數(shù)的聲明看起來很相似, 區(qū)別在于:

  • 在函數(shù)名字外面包了一層指針的語法
  • 函數(shù)的聲明是一條語句, 而函數(shù)指針是一個變量

當你定義了一個函數(shù)指針后, 這個指針變量的使用方法就像它指向的指針一樣, 只是換了一個名字.

int (*tester)(int a, int b) = sort_order; //注意 sort_order 不要帶括號
printf ("TEST: %d is same as %d\n", tester(2, 3), sort_order(2, 3));

如果 函數(shù)指針 指向的 函數(shù) 的 返回值 是 指針 的話, 是這個樣子的:

首先函數(shù)長這樣 : char *func(int a) {}
把函數(shù)名用指針包上: char *(*func)(int a)
然后改個名字 : char *(*p_func)(int a)

解決了函數(shù)指針的基本定義, 來看函數(shù)指針的另一個問題: 怎么把一個函數(shù)指針以參數(shù)的形式傳進另一個函數(shù)呢? 函數(shù)的參數(shù)都是有類型的, 那么首先要讓函數(shù)指針有自己的類型. 我們知道函數(shù)是由返回值和參數(shù)決定的, 那么函數(shù)指針的類型也是如此, 我們用 typedef 來定義一個函數(shù)指針的類型:

typedef int (*func_type) (int a, int b);

這樣, 就可以用 func_type 來作為函數(shù)的類型了, 這類型的函數(shù)返回值為一個 int, 參數(shù)為兩個 int.

我們來看下面一段代碼
這里主要看 typedef, 定義了一個函數(shù)的類型.
然后程序最后的 test_sorting, 傳入了三個函數(shù)指針.

#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>

/** Our old friend die from ex17. */
void die(const char *message)
{
    if(errno) {
        perror(message);
    } else {
        printf("ERROR: %s\n", message);
    }

    exit(1);
}

// a typedef creates a fake type, in this
// case for a function pointer
typedef int (*compare_cb)(int a, int b);

/**
 * A classic bubble sort function that uses the 
 * compare_cb to do the sorting. 
 */
int *bubble_sort(int *numbers, int count, compare_cb cmp)
{
    int temp = 0;
    int i = 0;
    int j = 0;
    int *target = malloc(count * sizeof(int));

    if(!target) die("Memory error.");

    memcpy(target, numbers, count * sizeof(int));

    for(i = 0; i < count; i++) {
        for(j = 0; j < count - 1; j++) {
            if(cmp(target[j], target[j+1]) > 0) {
                temp = target[j+1];
                target[j+1] = target[j];
                target[j] = temp;
            }
        }
    }

    return target;
}

int sorted_order(int a, int b)
{
    return a - b;
}

int reverse_order(int a, int b)
{
    return b - a;
}

int strange_order(int a, int b)
{
    if(a == 0 || b == 0) {
        return 0;
    } else {
        return a % b;
    }
}

/** 
 * Used to test that we are sorting things correctly
 * by doing the sort and printing it out.
 */
void test_sorting(int *numbers, int count, compare_cb cmp)
{
    int i = 0;
    int *sorted = bubble_sort(numbers, count, cmp);

    if(!sorted) die("Failed to sort as requested.");

    for(i = 0; i < count; i++) {
        printf("%d ", sorted[i]);
    }
    printf("\n");

    free(sorted);
}

int main(int argc, char *argv[])
{
    if(argc < 2) die("USAGE: ex18 4 3 1 5 6");

    int count = argc - 1;
    int i = 0;
    char **inputs = argv + 1;

    int *numbers = malloc(count * sizeof(int));
    if(!numbers) die("Memory error.");

    for(i = 0; i < count; i++) {
        numbers[i] = atoi(inputs[i]);
    }

    test_sorting(numbers, count, sorted_order);
    test_sorting(numbers, count, reverse_order);
    test_sorting(numbers, count, strange_order);

    free(numbers);

    return 0;
}
最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子烦却,更是在濱河造成了極大的恐慌轧坎,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,599評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機罐寨,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,629評論 3 385
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來汤纸,“玉大人衩茸,你說我怎么就攤上這事。” “怎么了楞慈?”我有些...
    開封第一講書人閱讀 158,084評論 0 348
  • 文/不壞的土叔 我叫張陵幔烛,是天一觀的道長。 經(jīng)常有香客問我囊蓝,道長饿悬,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,708評論 1 284
  • 正文 為了忘掉前任聚霜,我火速辦了婚禮狡恬,結果婚禮上,老公的妹妹穿的比我還像新娘蝎宇。我一直安慰自己弟劲,他們只是感情好,可當我...
    茶點故事閱讀 65,813評論 6 386
  • 文/花漫 我一把揭開白布姥芥。 她就那樣靜靜地躺著兔乞,像睡著了一般。 火紅的嫁衣襯著肌膚如雪凉唐。 梳的紋絲不亂的頭發(fā)上庸追,一...
    開封第一講書人閱讀 50,021評論 1 291
  • 那天,我揣著相機與錄音台囱,去河邊找鬼淡溯。 笑死,一個胖子當著我的面吹牛簿训,可吹牛的內(nèi)容都是我干的咱娶。 我是一名探鬼主播,決...
    沈念sama閱讀 39,120評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼煎楣,長吁一口氣:“原來是場噩夢啊……” “哼豺总!你這毒婦竟也來了?” 一聲冷哼從身側響起择懂,我...
    開封第一講書人閱讀 37,866評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎另玖,沒想到半個月后困曙,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,308評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡谦去,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,633評論 2 327
  • 正文 我和宋清朗相戀三年慷丽,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片鳄哭。...
    茶點故事閱讀 38,768評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡要糊,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出妆丘,到底是詐尸還是另有隱情锄俄,我是刑警寧澤局劲,帶...
    沈念sama閱讀 34,461評論 4 333
  • 正文 年R本政府宣布,位于F島的核電站奶赠,受9級特大地震影響鱼填,放射性物質發(fā)生泄漏。R本人自食惡果不足惜毅戈,卻給世界環(huán)境...
    茶點故事閱讀 40,094評論 3 317
  • 文/蒙蒙 一苹丸、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧苇经,春花似錦赘理、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,850評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至令花,卻和暖如春阻桅,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背兼都。 一陣腳步聲響...
    開封第一講書人閱讀 32,082評論 1 267
  • 我被黑心中介騙來泰國打工嫂沉, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人扮碧。 一個月前我還...
    沈念sama閱讀 46,571評論 2 362
  • 正文 我出身青樓趟章,卻偏偏與公主長得像,于是被迫代替她去往敵國和親慎王。 傳聞我的和親對象是個殘疾皇子蚓土,可洞房花燭夜當晚...
    茶點故事閱讀 43,666評論 2 350

推薦閱讀更多精彩內(nèi)容