欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C語(yǔ)言函數(shù)指針的老生常談

 更新時(shí)間:2021年11月24日 11:24:47   作者:青鋒楊  
這篇文章主要為大家介紹了vue組件通信的幾種方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助

函數(shù)指針

本質(zhì)上是一個(gè)指針,只不過指向函數(shù)而已。

編譯器在編譯期間對(duì)函數(shù)開辟了一塊空間,而這快空間的開始地址,就是它的函數(shù)指針 。

下面我們也直接用最直觀的程序來了解函數(shù)指針:

#if 1
void func()
{
	printf("hello ptr!");
}
int main()
{
	void (*p)();
	p = func;
	p();
}
#endif

在這里我們給出了func()函數(shù),并在其中打印hello ptr! 。在這里,我們定義了一個(gè)函數(shù)指針p,大家會(huì)發(fā)現(xiàn)他的定義語(yǔ)句十分的奇特: void (*p)() ,其實(shí)與int a,float b……十分的類似,定義嘛,實(shí)際上也就是給出類型,給出變量名,在這里void (*p)() 給出的類型是指向返回值為void 無(wú)參數(shù)傳入的函數(shù) ,在這里我們把這個(gè)指針命名為p。

這條程序中,我們直接將與p類型相對(duì)應(yīng)的函數(shù)func()的地址func賦值給p,于是我們便能拿著p去做func的事情了。

換言之,如果我們需要定義一個(gè)指向返回值為double,有一個(gè)int型參數(shù)a,一個(gè)long型參數(shù)b,名稱為f_ptr的函數(shù)指針呢?

那么顯然便是: double (*f_ptr)(int a,long b)

但其實(shí)因?yàn)槲覀冎恍枰嬖V編譯器我們的函數(shù)指針指向的是一個(gè)什么類型的函數(shù),所以參數(shù)名并不是必須的只需告訴他是什么類型即可,所以我們還可以簡(jiǎn)寫為 double (*f_ptr)(int , long)

我們還要注意到一個(gè)小細(xì)節(jié):

標(biāo)準(zhǔn)規(guī)定:函數(shù)名,可以認(rèn)為是其開始地址

所以函數(shù)指針p獲取函數(shù)地址: p = &Max; == p = Max;

函數(shù)指針p怎么調(diào)用: (*p)(10,20); == p(10,20);

我們同樣能夠拿上述程序來做個(gè)實(shí)驗(yàn):

函數(shù)指針的應(yīng)用

函數(shù)指針在我們程序中最常見的應(yīng)用其實(shí)應(yīng)該是作為函數(shù)的參數(shù)。比如,我們可以這樣:

int Add(int a, int b)  //加法實(shí)現(xiàn)
{
	return a + b;
}
int Sub(int a, int b)  //減法實(shí)現(xiàn)
{
	return a - b;
}
int Mul(int a, int b)  //乘法實(shí)現(xiàn)
{
	return a * b;
}
int Div(int a, int b)  //除法實(shí)現(xiàn)
{
	if (b == 0)
		return -1;
	return a / b;
}
int Computer(int a, int b, int(*p)(int, int))   //模板函數(shù)
{
	return p(a, b);
}
int main()
{
	printf("a+b=%d\n", Computer(10, 20, Add));
	printf("a-b=%d\n", Computer(10, 20, Sub));
	printf("a*b=%d\n", Computer(10, 20, Mul));
	printf("a/b=%d\n", Computer(10, 20, Div));
	return 0;
}

在這里,我們做出了一個(gè)模板函數(shù)computer,在他的參數(shù)列表中,我們給出了int (*p)(int,int)的參數(shù),我們調(diào)用函數(shù)的時(shí)候需要給到這個(gè)函數(shù)一個(gè)函數(shù)指針,而我們也在模板函數(shù)中使用了函數(shù)指針來調(diào)用具體函數(shù)。

我們定義了Add、Sub、Mul、Div四個(gè)具體的實(shí)現(xiàn)函數(shù),用來實(shí)現(xiàn)相應(yīng)的加減乘除,而computer只需要接受具體的函數(shù)指針來決定他需要干什么,這樣為程序的可拓展性提供了非常大的幫助,我們不需要在想要增刪功能的時(shí)候修改主要代碼,僅需增刪具體的功能函數(shù)就行,甚至部分庫(kù)函數(shù)利用函數(shù)指針作為參數(shù)的優(yōu)點(diǎn)來為我們提供可自定義的功能,下面也用一個(gè)例子來解釋。

函數(shù)指針作為參數(shù)實(shí)例(qsort函數(shù))

qsort函數(shù)包含在<stdlib.h>庫(kù)中

qsort函數(shù)的原型描述為:

void qsort( void *base, size_t n_elements, size_t el_sizeint ,int (*compar) (void const * , void const * ) )

qsort函數(shù)其實(shí)是C語(yǔ)言為我們編寫好的一個(gè)快速排序函數(shù),他通過函數(shù)指針為我們開放了一定的自定義功能

各參數(shù)解析,base中傳入需要排序的數(shù)組,n_elements為該數(shù)組中元素的個(gè)數(shù),el_sizeint為數(shù)組各元素的大小,而compar,則是我們今天反復(fù)提及的函數(shù)指針,在這里可以理解成為數(shù)組內(nèi)各元素的比較規(guī)則。

干說可能很難理解,直接上程序:

int compare_int(const void* a,const void* b)  //用于比較兩個(gè)整型值的具體函數(shù)
{
	//將函數(shù)傳入的參數(shù)進(jìn)行強(qiáng)轉(zhuǎn)
	int _a = *(const int*)a;
	int _b = *(const int*)b;
	//不相等
	if (_a > _b)
		return 1;
	else if (_a < _b)
		return -1;
	//相等
	else
		return 0;
}
int main()
{
	int arr[] = { 1,3,5,7,2,4,6,7,10,9,8 };
	qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof(int), compare_int);
	for (int i = 0; i < sizeof(arr)/sizeof(arr[0]); i++)
		printf("%d ", arr[i]);
	return 0;
}

此程序中,我們定義了自己的比較規(guī)則compare_int函數(shù),傳入到qsort函數(shù)的最后一個(gè)參數(shù),其余參數(shù)正常傳入,排序成功!

(qsort函數(shù)的比較規(guī)則默認(rèn)返回1則第一個(gè)參數(shù)大于第二個(gè)參數(shù),返回-1則反之,返回0則傳入的兩個(gè)參數(shù)相等)

在這里強(qiáng)轉(zhuǎn)是必須的,因?yàn)槟J(rèn)傳入的是兩個(gè)const void*類型,這在程序中是無(wú)法進(jìn)行比較的。

同時(shí),參數(shù)的類型也必須都寫為const void* 因?yàn)樵趒sort函數(shù)聲明中明確表示傳入的函數(shù)指針應(yīng)為返回值為int,兩個(gè)參數(shù)都為const void*類型,自定義比較規(guī)則函數(shù)的參數(shù)類型若不為const void*則會(huì)造成函數(shù)類型不匹配的問題。

我們?cè)谶@里也不妨想想一個(gè)字符串的指針數(shù)組應(yīng)該如何寫出我們的自定義函數(shù)來排序呢?

int compare_string(const void* _str1, const void* _str2)
{
	//強(qiáng)轉(zhuǎn)類型
	const char* str1 = *(const char**)_str1;
	const char* str2 = *(const char**)_str2;
	//利用string.h庫(kù)中stramp函數(shù)(返回值契合qsort函數(shù)比較規(guī)則)來進(jìn)行比較
	int tem = strcmp(str1, str2);
	return tem;
}
int main()
{
	const char* arr[] = { "abc","dsa","adfw","odc","adsfa","afsd" };
	qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof(const char*), compare_string);
	for (int i = 0; i < sizeof(arr) / sizeof(arr[0]);i++)
		printf("%s ", arr[i]);
	return 0;
}

可能在這大家疑惑的地方在于為什么需要使用一個(gè)二級(jí)指針來進(jìn)行強(qiáng)轉(zhuǎn),其實(shí)在這里可以用我們平時(shí)的指針參數(shù)傳入來理解,比如void func(int* a),我們傳入給這個(gè)帶*參數(shù)時(shí)實(shí)際上傳入的是a的地址。

同理我們傳入_str的是什么?我們?cè)瓟?shù)組是一個(gè)指針數(shù)組,每個(gè)元素本質(zhì)上是個(gè)一級(jí)指針,那么我們每次比較時(shí)傳入給compare_string函數(shù)一個(gè)帶*的參數(shù),是不是那便是一級(jí)指針的地址,即二級(jí)指針,加上一個(gè)解引用的*變回一級(jí)指針便可以賦給str1了。

總結(jié)

本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!

相關(guān)文章

  • Opencv透視變換綜合實(shí)例詳解

    Opencv透視變換綜合實(shí)例詳解

    這篇文章主要為大家詳細(xì)介紹了Opencv透視變換綜合實(shí)例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-05-05
  • 詳解C++編程中的虛函數(shù)

    詳解C++編程中的虛函數(shù)

    這篇文章主要介紹了詳解C++編程中的虛函數(shù),包括在什么情況下應(yīng)當(dāng)聲明虛函數(shù)的相關(guān)講解,需要的朋友可以參考下
    2015-09-09
  • C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)哈希表詳解

    C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)哈希表詳解

    哈希表是一種根據(jù)關(guān)鍵碼去尋找值的數(shù)據(jù)映射結(jié)構(gòu),該結(jié)構(gòu)通過把關(guān)鍵碼映射的位置去尋找存放值的地方,說起來可能感覺有點(diǎn)復(fù)雜,我想我舉個(gè)例子你就會(huì)明白了,最典型的的例子就是字典
    2022-02-02
  • C語(yǔ)言對(duì)結(jié)構(gòu)體數(shù)組按照某項(xiàng)規(guī)則進(jìn)行排序的實(shí)現(xiàn)過程探究

    C語(yǔ)言對(duì)結(jié)構(gòu)體數(shù)組按照某項(xiàng)規(guī)則進(jìn)行排序的實(shí)現(xiàn)過程探究

    這篇文章主要介紹了C語(yǔ)言對(duì)結(jié)構(gòu)體數(shù)組按照某項(xiàng)規(guī)則進(jìn)行排序的實(shí)現(xiàn)過程,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧
    2023-02-02
  • VisualStudio類文件的管理(類文件的分離)的實(shí)現(xiàn)

    VisualStudio類文件的管理(類文件的分離)的實(shí)現(xiàn)

    在使用?Visual?Studio?開發(fā)項(xiàng)目的時(shí)候,學(xué)會(huì)進(jìn)行“類文件的分離”十分重要,本文主要介紹了VisualStudio類文件的管理(類文件的分離)的實(shí)現(xiàn),感興趣的可以了解一下
    2024-03-03
  • C++泛型編程基本概念詳解

    C++泛型編程基本概念詳解

    這一篇介紹一下 C++ 編程中與面向?qū)ο蟛⒘械牧硪淮蠓种А盒途幊?,這一篇主要介紹函數(shù)模板、類模板和成員模板三大部分,需要的朋友可以參考下
    2021-08-08
  • C++深入講解new與deleted關(guān)鍵字的使用

    C++深入講解new與deleted關(guān)鍵字的使用

    這篇文章主要介紹了C++中new與deleted關(guān)鍵字的使用,new在動(dòng)態(tài)內(nèi)存中為對(duì)象分配空間并返回一個(gè)指向該對(duì)象的指針;delete接受一個(gè)動(dòng)態(tài)對(duì)象的指針, 銷毀該對(duì)象, 并釋放與之關(guān)聯(lián)的內(nèi)存
    2022-05-05
  • Qt 進(jìn)度條的實(shí)現(xiàn)示例

    Qt 進(jìn)度條的實(shí)現(xiàn)示例

    進(jìn)度條在很多時(shí)候都可以用到,有時(shí)我們需要在表格,樹狀欄中直觀顯示任務(wù)進(jìn)度或消耗百分比,本文就詳細(xì)的介紹一下Qt 進(jìn)度條的使用實(shí)例,感興趣的可以了解一下
    2021-06-06
  • C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之二叉樹詳解

    C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之二叉樹詳解

    二叉樹(Binary tree)是樹形結(jié)構(gòu)的一個(gè)重要類型。許多實(shí)際問題抽象出來的數(shù)據(jù)結(jié)構(gòu)往往是二叉樹形式。本文將通過示例詳細(xì)講解一下二叉樹,需要的可以參考一下
    2022-03-03
  • C++的對(duì)象特性和友元你真的了解嗎

    C++的對(duì)象特性和友元你真的了解嗎

    這篇文章主要為大家詳細(xì)介紹了C++的對(duì)象特性和友元,使用數(shù)據(jù)庫(kù),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-02-02

最新評(píng)論