1.知识点
(1)在程序中可以声明指向任何数据类型的指针,指针也可以指向指针类型,成为指向指针的指针。下面是一个使用例子
1 int a=10,b=20;
2 int *q=&a;
3 int **p=&q;
4 **p=30;
(2)如果想通过指针在被调函数中修改主调函数的变量,必须将主调函变量(务必确定该变量的类型,有时候可能变量本身就是指针,这时候形参就需要是指针的指针了)的地址作为参数,在被调函数中修改主调函数指向的内容。
2.面试题
2.1指针作为参数的常见错误
1 int find(char *s, char ch, char *sub) {
2 for (int i = 0; *(s + i) != '\n'; i++) {
3 if (*(s + i) == ch) {
4 sub = s + i + 1;
5 return i;
6 }
7 }
8 return 0;
9 }
10
11 int main(int argc, char *argv[]) {
12 char fullName[] = {"Jordan#Michale"};
13 char *givenName=fullName;
14
15 int cnt = find(fullName, '#', givenName);
16 cout << givenName << "has a" << cnt << "characters 'family name" << endl;
17
18 getchar();
19 return 0;
20 }
重点知识点:(1)如果想通过指针在被调函数中修改主调函数的变量,必须将主调函变量的地址作为参数,在被调函数中修改主调函数指向的内容。
(2)通过指针传递参数时,最大的忌讳就是以为只要参数是指针就万事大吉了。实际上,应该首先确定要修改的变量的类型,然后再将其地址作为参数。如果要修改的变量本身就是指针,就应该将该指针的地址作为参数,此时的形参类型是指向指针的指针。
以上第三个参数的指向的改变并不能带来实参的改变,正确答案如下:
1 int find(char *s, char ch, char **sub) {
2 for (int i = 0; *(s + i) != '\n'; i++) {
3 if (*(s + i) == ch) {
4 *sub = s + i + 1;
5 return i;
6 }
7 }
8 return 0;
9 }
10
11 int main(int argc, char *argv[]) {
12
13 char fullName[] = {"Jordan#Michale"};
14 char *givenName=NULL;
15
16 int cnt = find(fullName, '#', &givenName);
17 cout << givenName << " has a " << cnt << " characters 'family name" << endl;
18
19 getchar();
20 return 0;
21 }
2.2指向指针的指针和二维数组的区别
1 int main(){
2 int a[2][3]={{1,2,3},{4,5,6}};
3 int **p=a;
4 cout<<**p<<endl;
5 getchar();
6 return 0;
7 }
解析:p是指向指针的指针(类型是int **),a<=>&a[0],a[0]是一个一维数组(相当于对一维数组名a[0]取地址,它应该赋值给int *[3])。所以左边是一个指向指针的指针,右边是一个指向数组的指针,两边类型不同。如果想编译通过,应该定义一个指向数组的指针,修改如下:
1 int (*p)[3]=a;
小技巧:判断变量的类型:
变量的类型在声明之初就已经确定了,在程序中只要将声明语句中变量名去掉,剩下的部分就是变量的类型。如下:
1 int **p的类型是:int**
2 int a[2][3]的类型是:int[2][3]
3 const int(const *p)[3]的类型是:const int(cosnt *)[3]