C語言精髓是指針,指針知識深似海,遇到一些學習一些~
文章目錄
- 1. typedef 定義函數指針類型
- 2. void* 空指針的解引用
1. typedef 定義函數指針類型
函數參數化是指通過函數指針將函數的某些行為參數化。這樣,我們可以在調用函數時動態地指定函數的行為。以下是一個示例:
#include <stdio.h>void process_array(int *array, size_t size, int (*process)(int))
{for(size_t i = 0; i < size; i++){array[i] = process(array[i]);}
}int increment(int n)
{return n * 2;
}int main()
{int array[] = {1, 2, 3, 4, 5};size_t size = sizeof(array) / sizeof(int);process_array(array, size, increment);for(size_t i =0; i < size; i++){printf("%d", array[i]);}printf("\n");return 0;
}
如果通過typedef來定義函數指針類型:
#include <stdio.h>typedef int (*process_t)(int);void process_array(int *array, size_t size, process_t process)
{for(size_t i = 0; i < size; i++){array[i] = process(array[i]);}
}int increment(int n)
{return n * 2;
}int main()
{int array[] = {1, 2, 3, 4, 5};size_t size = sizeof(array) / sizeof(int);process_array(array, size, increment);for(size_t i =0; i < size; i++){printf("%d", array[i]);}printf("\n");return 0;
}
定義的 process_array 函數接受三個參數:一個整型數組、數組大小、一個函數指針。函數指針指向一個函數,該函數接受一個整型參數并返回一個整型結果。因此,當執行到 array[i] = process(array[i]);
時,process函數會自動調用increment函數。
有人可能會問了:“那如果我再定義一個函數:int aaaa(int x){return x + 1;}呢?process 函數會調用哪個函數?”。
這樣符合 process_t 類型的函數就有多個了,我們可以將 process 指針指向某個函數,比如:
process_t process = aaaa; // 設置 process 指針指向 aaaa 函數
process_array(array, size, process);
2. void* 空指針的解引用
首先定義一個uint32_t 類型的param_value ,其值為0xa5a5,現在已經通過某種方法將 param_value 的地址指針傳入了函數 sys_manager_handler 中(第三個參數),我們需要關注的是如何在函數內使用printf將 param_value 這個值打印出來。
在printf中,應該使用何種占位符?param作為一個指針又應該如何變換?
偽代碼如下:
uint32_t param_value = 0xa5a5;int sys_manager_handler(void *cobj, uint32_t event_id, void *param)
{printf("we got value from event:%lx", *(uint32_t *)param);printf("sys_manger got value from event:%lx", *param);
}
以上兩句 printf 哪個正確?
由于我們知道,傳入函數的第三個參數 param 是一個指向 param_value 地址的指針,所以我們在 printf 中的變量處需要對 param 指針進行解引用,得到一個 uint32_t 類型的變量,對應的占位符使用 %lx 就好啦。
但其實直接對 param 進行解引用是不對的,因為:
- void * 類型的指針可以指向任何類型的數據,對應地,這樣的指針并沒有具體的類型信息,所以編譯器也不知道它指向的數據的類型和大小。
- 當需要訪問void *指針所指向的數據時,必須先將其轉換為具體的指針類型,然后才能解引用。
綜上,正確的是第一句。
printf("we got value from event:%lx", *(uint32_t *)param);