c語言中的二級指針做函數(shù)參數(shù)說明
二級指針做函數(shù)參數(shù)
1.用指針做函數(shù)參數(shù)申請動態(tài)內(nèi)存的問題
//如果函數(shù)參數(shù)是指針,不能用一級指針做函數(shù)參數(shù)實(shí)現(xiàn)申請動態(tài)內(nèi)存 ? void getMemory(char *p, int num) { ?? ?p = (char *)malloc(sizeof(char)*num); } void main() { ?? ?char *str = NULL; ?? ?getMemory(str, 20); ?? ?strcpy(str, "code"); ?? ?cout << str; ?? ?system("pause"); }//出錯(cuò) 然而 //用二級指針為指針申請內(nèi)存,用二級指針做函數(shù)參數(shù)可以申請成功 void getMemory1(char **p, int num) { ?? ?*p = (char *)malloc(sizeof(char)*num); } void main() { ?? ?char *str = NULL; ?? ?getMemory(&str); ?? ?strcpy(str, "code"); ?? ?cout << str; ?? ?system("pause"); }
2.二級指針做函數(shù)參數(shù)簡介修改變量的值
#include<iostream> ? using namespace std; struct Teacher { ?? ?char name[20]; ?? ?int age; }; int getTeacher(Teacher **p) { ?? ?Teacher *tmp = NULL; ?? ?if (p == NULL) ?? ?{ ?? ??? ?return -1; ?? ?} ?? ?if (tmp == NULL) ?? ?{ ?? ??? ?return -2; ?? ?} ?? ?tmp->age = 23; ?? ?*p = tmp;//p是實(shí)參地址,*p間接修改實(shí)參的值 ?? ?//用二級指針修改一級指針的值,用一級指針修改零級指針的值 } void FreeTeacher(Teacher *p) { ?? ?if (p == NULL) ?? ?{ ?? ??? ?return; ?? ?} ?? ?free(p); } int main() { ?? ?Teacher *pt = NULL; ?? ?getTeacher(&pt); ?? ?cout << pt->age << endl; ?? ?FreeTeacher(pt); ?? ?system("pause"); ?? ?return 0; }
二級指針作為形參簡單實(shí)例分析
指針是什么?指針是程序數(shù)據(jù)在內(nèi)存中的地址,而指針變量是用來保存這些地址的變量。
舉例
int c=2,d=3; int *pc=&c;
pc是指針變量的存儲內(nèi)容,也就是c的地址
*pc就是對指針的解引用,取出這個(gè)c這個(gè)地址里面的值
實(shí)例
#include <stdio.h> int c=2,d=3; int *pc=&c; //一級指針指向c int **ppc = &pc; //二級指針指向一級指針pc (指向指針的指針) int main() { //一級指針*p指向c printf("一級指針*p指向c\n\n"); printf("c的地址=%p c的地址\n",&c); printf("pc的值 =%p pc中指向的就是C的地址\n",pc); printf("*pc的值=%d 對c的地址進(jìn)行取值*pc==2;\n",*pc); putchar('\n'); printf("-------------------------------------------------------------\n"); //二級指針(指針的指針)**ppc指向*pc printf("二級指針(指針的指針)**ppc指向*pc\n\n"); printf("pc的地址 =%p 一級指針pc的地址\n",&pc); printf("ppc的值 =%p 二級指針ppc指向的地址其實(shí)就是一級指針pc的地址\n",ppc); printf("*ppc的值 =%p 對ppc進(jìn)行一次解引用,*ppc==pc,*ppc也就是c的地址\n",*ppc); printf("*(*ppc)的值=%d 對ppc進(jìn)行兩次解引用,*(*ppc)==*pc==c==2\n",*(*ppc)); putchar('\n'); return 0; }
運(yùn)行結(jié)果為:
一級指針二級指針做函數(shù)形參
許多初學(xué)者在使用一級指針的時(shí)候基本沒什么問題
例如
int a=10; int fun(int *p) { ? ? ? ? ? return (?*p+=10); }
在調(diào)用是時(shí)需要定義一個(gè)變量int b=fun(&a);
當(dāng)二級指針做形參的時(shí)候,調(diào)用該函數(shù)時(shí)實(shí)參應(yīng)該以什么格式填寫?
int a=1; int b=2; int fun1(int **p1,int *p2)//二級指針做形參 后面舉例用 { return (**p1+*p2); } int fun2(int x,int y) { return (x+y); }
函數(shù)fun1的第一個(gè)形參是**p1 二級指針 , 第二個(gè)是*p2 , 最終的目的是將兩個(gè)變量相加
我們對比fun2, 這個(gè)函數(shù)的形參都是普通變量,調(diào)用時(shí)直接int sum=fun2(a,b),最終的目的是將兩個(gè)變量相加.
通過對比,我們發(fā)現(xiàn)**p1代表一個(gè)變量,*p2也代表一個(gè)變量。
我們來分析下*p這個(gè)東西。
int a=2; int b=3; int *p=&a;
我們可以知道 p存放的值就是變量a的地址,*(地址)就相當(dāng)于把這個(gè)地址的值取出來。
假如a的地址等于0x123456, p也==0x123456, b=(*p)==*(a的地址)==a的值;通過這步b就等于2;
我們回到fun1函數(shù),**p怎么才能轉(zhuǎn)換 為一個(gè)變量呢,看下圖
一級指針只有一層解引用——*(地址)則得到一個(gè)變量,
二級指針內(nèi)層解引用得到一個(gè)地址add2,對add2解引用得到一個(gè)變量
*(add1)=add2,*(add2)=變量 ——> *(*(add1))=變量
所以fun1的第一個(gè)實(shí)參應(yīng)該填一個(gè)地址ADD1 對ADD1解引用得到ADD2,對ADD2解引用得到一個(gè)變量
下面看看實(shí)例
#include <stdio.h> int c=2,d=3; int *pc=&c; int **ppc = &pc; int fun1(int **p1,int *p2)//二級指針做形參 后面舉例用 { return (**p1+*p2); } int main() { int sum=fun1(&pc,&d);//在二級指針形參填寫一個(gè)指針的地址,這個(gè)指針指向一個(gè)變量 printf("sum=%d\n",sum); putchar('\n'); printf("------------------------------------------\n"); printf(" c的地址=%p\n",&c); printf(" pc指向的地址=%p\n",pc); printf(" *pc=%d\n",*pc); putchar('\n'); printf("------------------------------------------\n"); printf(" c的地址=%p\n",&c); printf(" pc指向的地址=%p\n",pc); printf(" pc的地址=%p\n",&pc); putchar('\n'); printf("------------------------------------------\n"); printf(" ppc指向的地址=%p\n",ppc); printf(" *(ppc)的地址=%p\n",*(ppc)); printf(" **(ppc)的值=%d\n",**(ppc)); putchar('\n'); return 0; }
解析一下程序二級指針的流程
三級指針或者多級指針以此類推,凡是涉及到指針解引用的*(x),x都是一個(gè)地址。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
簡述C++中虛擬函數(shù)的內(nèi)存分配機(jī)制
這篇文章主要介紹了簡述C++中虛擬函數(shù)的內(nèi)存分配機(jī)制,幫助大家更好的理解和學(xué)習(xí)c++,感興趣的朋友可以了解下2020-08-08C語言實(shí)現(xiàn)父進(jìn)程主動終止子進(jìn)程的方法總結(jié)
一般的情況,子進(jìn)程自己運(yùn)行完后,執(zhí)行exit 或者return 后,父進(jìn)程wait. waitpid收回子進(jìn)程,但子進(jìn)程是一個(gè)循環(huán)等待狀態(tài)不主動退出,父進(jìn)程可以采用文中介紹的幾種方法,需要的朋友可以參考下2023-10-10C++中的不規(guī)則二維數(shù)組實(shí)現(xiàn)代碼
本文介紹了一個(gè)在C++中保存不定長二維數(shù)組的數(shù)據(jù)結(jié)構(gòu),在這個(gè)結(jié)構(gòu)中,我們使用了一個(gè)含有指針和數(shù)組長度的結(jié)構(gòu)體,用這樣的一個(gè)結(jié)構(gòu)體構(gòu)造一個(gè)結(jié)構(gòu)體數(shù)組,用于存儲每一個(gè)不定長的數(shù)組,感興趣的朋友一起看看吧2024-03-03C++深淺拷貝及簡易string類實(shí)現(xiàn)方式
這篇文章主要介紹了C++深淺拷貝及簡易string類實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-02-02QT網(wǎng)絡(luò)編程UDP下C/S架構(gòu)廣播通信(實(shí)例講解)
下面小編就為大家?guī)硪黄猀T網(wǎng)絡(luò)編程UDP下C/S架構(gòu)廣播通信(實(shí)例講解)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-07-07