C語(yǔ)言進(jìn)階教程之函數(shù)指針詳解
一、函數(shù)指針
1.概念
函數(shù)指針:首先它是一個(gè)指針,一個(gè)指向函數(shù)的指針,在內(nèi)存空間中存放的是函數(shù)的地址;
請(qǐng)看示例:
int main(){
int a = 10;
int*pa = &a;
char ch = 'c';
char* pc = &ch;
int arr[10] = {0};
int (*parr)[10] = &arr;//取出數(shù)組的地址
return 0;
}
解析:parr是一個(gè)指向數(shù)組的指針,存放的是數(shù)組的地址;
所以:
- 數(shù)組指針 —存放數(shù)組地址的指針;
- &數(shù)組名 —得到的就是數(shù)組的地址;
那么我們可以不可以這么認(rèn)為:
- 函數(shù)指針 —存放函數(shù)地址的指針;
- &函數(shù)名 —得到的就是一個(gè)函數(shù)的地址;
是這樣嗎?我們來(lái)測(cè)試一下,請(qǐng)看下面事例:
int Add(int x,int y)
{
return x+y;
}
int main()
{
printf("%p\n",&Add);//打印一下函數(shù)Add()的地址
printf("%p\n",Add);//數(shù)組名等于數(shù)組首元素地址,那函數(shù)名是等于函數(shù)地址嗎?
return 0;
}
請(qǐng)看結(jié)果:
哦!原來(lái),函數(shù)名是等于函數(shù)地址的!
1.2函數(shù)指針的使用方法
函數(shù)指針的定義: 函數(shù)的返回值類型(*指針名)(函數(shù)的參數(shù)列表類型)
int Add(int x, int y)
{
return x+y;
}
int main()
{
int (*pf)(int, int) = &Add;//函數(shù)指針定義,返回值類型和參數(shù)類型與函數(shù)Add()相同
}
1.3練習(xí)鞏固
void test(char* str){}
int main (){
①(?)pt =&test;
return 0;
}
請(qǐng)問(wèn)①語(yǔ)句應(yīng)該怎么完善呢?
答案:void ( * pt)(char*) = &test;
怎么使用函數(shù)指針去調(diào)用函數(shù)呢?
還是上面的例子:
void Add(int x, int y){
return x+y;}
int main(){
int (*pf)(int,int)=&Add;
int ret=(*pf)(3,5);
解析:
int ret=(*pf)(3,5),此時(shí)就相當(dāng)于通過(guò)函數(shù)名調(diào)用: int ret=Add(3,5);},我們又知道:函數(shù)名是等于&函數(shù)名的,所以int (*pf)(int,int)=&Add,可改成:int (*pf)(int,int)=Add;此時(shí)Add等價(jià)于pf,所以:int ret=(*pf)(3,5);語(yǔ)句可改成:int ret=pf(3,5);等價(jià)于int ret=Add(3,5),故我們知道了對(duì)于:int ret=(*pf)(3,5);語(yǔ)句來(lái)說(shuō),*是沒(méi)有意義的,有一個(gè)或多個(gè)或者沒(méi)有都不影響;
1.4小結(jié)一下
數(shù)組名 (arr) != &數(shù)組名(&arr)
函數(shù)名(Add) = &函數(shù)名(&Add)
二、閱讀兩段有趣的代碼
注:來(lái)源于《c陷阱和缺陷》;
1.( *(void( *)( ))0 )( )
解析:
調(diào)用0地址處的函數(shù)該函數(shù)無(wú)參數(shù),返回值是void拆分:
這段代碼的含義是:
●void()() 表示函數(shù)指針類型
●( void()() )0 表示對(duì)0進(jìn)行強(qiáng)制類型轉(zhuǎn)換,把0強(qiáng)制類型轉(zhuǎn)換成一個(gè)函數(shù)的地址;如(int)3.14
●* ( void()() )0 表示對(duì)0地址處的函數(shù)進(jìn)行了解引用操作
●( ( void(*)() )0)() 則表示調(diào)用0地址處的函數(shù)
請(qǐng)看圖解:
2.void (* signal(int,void( * )( int ) ) )(int)
解析:
1.signal和()先結(jié)合,說(shuō)明signal是一個(gè)函數(shù)名
2.signal函數(shù)第一個(gè)參數(shù)的類型為int,第二個(gè)參數(shù)的類型為函數(shù)指針,該函數(shù)指針指向一個(gè)參數(shù)為int,返回值為void的函數(shù);
3.signal 函數(shù)的返回類型也是一個(gè)函數(shù)指針,該函數(shù)指針,指向一個(gè)參數(shù)為int,返回值為void函數(shù)
4.請(qǐng)看圖解:
綜上,signal是一個(gè)函數(shù)聲明;
附:函數(shù)指針的應(yīng)用——函數(shù)回調(diào)
#include<stdio.h>
#include<stdlib.h>
#include<time.h>
#define arrLen 10
void selectSort(int *p,int n,int (*pf)(int ,int )){
for(int i = 0;i<n-1;i++){
for(int j=i+1;j<n;j++){
if(pf(p[i],p[j])){
p[i] = p[i]^p[j];
p[j] = p[i]^p[j];
p[i] = p[i]^p[j];
}
}
}
}
//只需修改啊a>B 或a<b,不用修改 selectSort就可以實(shí)現(xiàn)降序升序排列
//這就是函數(shù)回調(diào)帶來(lái)的便利
//這也是函數(shù)指針一種應(yīng)用
int callBack(int a,int b){
return a<b?1:0;
}
int main(void){
srand(time(NULL));
int arr[arrLen] ;
for(int i = 0;i<arrLen;i++){
arr[i] = rand()%100+1;
}
selectSort(arr,arrLen,callBack);
for(int i = 0;i<arrLen;i++){
printf("%5d",arr[i]);
}
return 0;
}
總結(jié)
到此這篇關(guān)于C語(yǔ)言進(jìn)階教程之函數(shù)指針的文章就介紹到這了,更多相關(guān)C語(yǔ)言函數(shù)指針內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語(yǔ)言中的時(shí)間函數(shù)clock()和time()你都了解嗎
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言中的時(shí)間函數(shù)clock()和time(),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-02-02
解析如何用指針實(shí)現(xiàn)整型數(shù)據(jù)的加法
本篇文章是對(duì)用指針實(shí)現(xiàn)整型數(shù)據(jù)加法的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
C語(yǔ)言實(shí)現(xiàn)大數(shù)值金額大寫轉(zhuǎn)換的方法詳解
這篇文章主要為大家詳細(xì)介紹了如何利用C語(yǔ)言實(shí)現(xiàn)大數(shù)值金額大寫轉(zhuǎn)換的功能,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起了解一下2023-03-03
C++ Eigen庫(kù)實(shí)現(xiàn)最小二乘擬合的示例代碼
Eigen 是一個(gè)線性算術(shù)的 C++ 模板庫(kù),功能強(qiáng)大、快速、優(yōu)雅以及支持多平臺(tái),本文主要為大家介紹了C++利用Eigen庫(kù)實(shí)現(xiàn)最小二乘擬合的示例代碼,希望對(duì)大家有所幫助2023-07-07
解析C++中臨時(shí)對(duì)象的產(chǎn)生情況
臨時(shí)對(duì)象的產(chǎn)生和銷毀都是有成本的,都會(huì)影響程序的執(zhí)行性能和效率,所以如果能了解臨時(shí)對(duì)象產(chǎn)生的原因,就可以提升程序的性能和效率,下面小編就來(lái)和大家聊聊臨時(shí)對(duì)象產(chǎn)生的幾種情況吧2023-06-06
C語(yǔ)言經(jīng)典算法例題求100-999之間的“水仙花數(shù)”
本文的主要內(nèi)容,設(shè)計(jì)一個(gè)程序,找出100-999之間的“水仙花數(shù)”,需要的朋友可以參考下2015-07-07
使用kendynet構(gòu)建異步redis訪問(wèn)服務(wù)
這篇文章主要介紹了在kendynet上寫的一個(gè)簡(jiǎn)單的redis異步訪問(wèn)接口,大家參考使用吧2014-01-01
C++使用GDAL庫(kù)實(shí)現(xiàn)Tiff文件的讀取
這篇文章主要為大家詳細(xì)介紹了C++使用GDAL庫(kù)實(shí)現(xiàn)Tiff文件的讀取的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-03-03




