2,指向指針的指針,下面分別看幾個(gè)例子
char string[3]={'c','d','e'}.定義了三個(gè)char型的變量string[0]='c',string[1]='d',string[2]='e'.
char *pstring={"cde"},定義了一個(gè)字符串,其中string[0]='c',string[1]='d',string[2]='e'.
現在明顯可以看出上面兩者的聯(lián)系了。其實(shí)string=pstring.
char string[3][3]={'M','o','n';'T','u','e';'W','e','d'};這樣string[0][0]='M',,,,,
char *pstring[3]={"Mon","Tue","Wed"},定義了三個(gè)char *string型的變量。string[0]="Mon",string[0][0]='M'
char **ppstring.那這個(gè)是什么呢,他就是個(gè)指向指針的指針,可以直接初始化ppstring=pstring,或者可以ppstring=&(pstring[0]),使用的時(shí)候可以完全和*pstring[3]差不多使用了。
所以指向指針的指針也就是多了一個(gè)間接訪(fǎng)問(wèn)而已,只要知道了指針就是地址,*就是去訪(fǎng)問(wèn)地址的數據就可以了。
3,函數指針。
int (*f)(),前面一個(gè)括號迫使間接訪(fǎng)問(wèn)在函數調用之前,那么就是說(shuō)訪(fǎng)問(wèn)內存中的某個(gè)位置的函數。這就是函數指針的概念。
初始化:int f(int); int (*pf)(inf)=&f;
調用方式:int ans=pf(25);
下面討論兩個(gè)具體應用的問(wèn)題:
1),回調函數
我們先看看下面這段程序:
int comp(int a,int b)
{
if(a==b)return 0;
else return 1;
}
完成的功能實(shí)在是簡(jiǎn)單,就是比較一下兩個(gè)數是否一致。但是你是否發(fā)現這個(gè)只能比較int型的數據,要是比較char的呢,那必須得重新寫(xiě)一個(gè)函數,比較float也得重新寫(xiě)一個(gè),這樣是不是有點(diǎn)麻煩,并且易用性也不好,換一個(gè)類(lèi)型就要換一個(gè)函數。那么我們考慮是否在comp的時(shí)候不傳具體的類(lèi)型呢,因為我們根本也不知道要傳什么類(lèi)型給comp。解決難題的辦法就是把參數類(lèi)型聲明為void *,表示"一個(gè)指向未知類(lèi)型的指針"。然后加一個(gè)回調函數作為參數,這樣在按照不同的類(lèi)型編寫(xiě)不同的回調函數,在上層就保證了comp函數的易用性。下面請看改寫(xiě)的函數
int comp(void *a,void *b, int (*compare)(void const *,void const *))
{
if(compare(a,b))return 1;
else return 0;
}
int compare_int(void const *a,void const *b)
{
if(*(int*)a==*(int*)b) return 0;
else return 1;
}
int compare_char(void const *a,void const *b)
{
if(*(char*)a==*(char*)b) return 0;
else return 1;
}
在主函數中可以這樣調用
int a=2;int b=2;
comp(&a,&b,compare_int);
char c='d';char d='d';
comp(&d,&c,compare_char);
現在在C++中已經(jīng)有重載和模板的技術(shù)了,上面的任務(wù)根本不用這么麻煩實(shí)現的。但是openssl是用C語(yǔ)言寫(xiě)的,里面大量的void類(lèi)型,并且大量的指向函數的指針,希望這點(diǎn)講解對大家看openssl能有點(diǎn)幫助。
2),轉移表
我們下面看個(gè)例子,在一個(gè)計算器的例子中,有如下一些語(yǔ)句:
switch(oper)
case ADD:
result=add(op1,op2);
break;
case SUB:
result=sub(op1,op2);
break;
case MUL:
result=mul(op1,op2);
break;
case DIV:
result=div(op1,op2);
break;
...
對于一個(gè)功能復雜的計算器,那么switch語(yǔ)句將非常長(cháng)。為了使用switch語(yǔ)句,表示操作符的代碼必須是整數。如果它們是從零開(kāi)始連續的整數,我們可以使用轉換表來(lái)完成這個(gè)任務(wù)。轉換表就是一個(gè)函數指針數組。
創(chuàng )建一個(gè)轉換表需要兩個(gè)步驟。首先,聲明并初始化一個(gè)函數指針數組。唯一需要留心之處就是確保這些函數的原形出現在這個(gè)數組的聲明之前。
double add(double,double);
double sub(double,double);
double mul(double,double);
...
double (*oper_func[])(double,double)={add,sub,mul,div,...};
初始化列表中各個(gè)函數名的正確順序取決于程序中用于表示每個(gè)操作符的整型代碼。這個(gè)例子假定ADD是0,SUB是1,MUL是3,,,
第2個(gè)步驟是用下面語(yǔ)句替換前面整條switch語(yǔ)句:
result=oper_func[oper](op1,op2);
oper從數組中選擇正確的函數指針,而函數調用操作符將執行這個(gè)函數。
聯(lián)系客服