【C语言】进阶指针(一)

简介: 【C语言】进阶指针(一)

前言:

进阶指针我打算分三篇文章进行讲解,第一篇主要围绕数组与指针进行,第二篇主要围绕函数指针进行,最后一篇会引入面试题讲解。

指针的基本概念:

1、指针就是个变量,用来存放地址,地址唯一标识一块内存空间。

2、指针的大小是固定的4/8个字节(32位/64位)

3、指针是有类型,指针的类型决定的指针+-整数的步长,指针解引用操作的权限。

一、字符指针

字符指针就是指针类型为字符的指针:char*。

可以这样定义:

int main()
{
  char ch = 'w';
  char* pc = &ch;
  *pc = 'w';
  return 0;
}

如果是字符串呢?

int main()
{
  char* pstr = "hello world.";
  printf("%s\n", pstr);
  return 0;
}

注意这里指针变量pstr中存放的是字符串中首字符的地址

一道面试题:

int main()
{
    char str1[] = "hello world.";
    char str2[] = "hello world.";
    const char* str3 = "hello world.";
    const char* str4 = "hello world.";
    if (str1 == str2)
        printf("str1 and str2 are same\n");
    else
        printf("str1 and str2 are not same\n");
    if (str3 == str4)
        printf("str3 and str4 are same\n");
    else
        printf("str3 and str4 are not same\n");
    return 0;
}

输出结果:

str1 and str2 are not same

str3 and str4 are same

分析:str1与str2为两个数组,而数组名代表首元素地址,所以str1与str2不相同。而str3与str4指向的是同一个字符串常量,当几个指针指向同一个字符串的时候,他们实际会指向同一块内存

二、指针数组与数组指针

(一)指针数组

指针数组是数组,数组存放的内容是指针,那么很简单定义数组的时候,数组类型定义为XXX*即可,如:

int* arr1[10]; //整型指针的数组
char* arr2[4]; //一级字符指针的数组
char** arr3[5];//二级字符指针的数组

(二)数组指针

数组指针是指针,指针指向的内容为数组。

观察下面的代码,哪个是数组指针?

int* p1[10];
int (*p2)[10];

注:[]操作符的优先级高于*。

分析:p1为指针数组。p2先于*结合,即p2为指针变量,指针变量的类型为int [10]整型数组,所以p2为数组指针。

我们知道数组名arr表示首元素的地址,那么&arr 取出的是不是首元素的地址呢?

其实数组名表示数组首元素的地址有两个例外:

1、sizeof(arr),sizeof计算的是整个数组大小,单位为字节。

2、&arr,此时取出的是整个数组的地址。

我们利用下面的代码进行验证:

int main()
{
  int arr[10] = { 0 };
  printf("arr = %p\n", arr);
  printf("&arr= %p\n", &arr);
  printf("arr+1 = %p\n", arr + 1);
  printf("&arr+1= %p\n", &arr + 1);
  return 0;
}

结果为:

我们发现&arr+1跳过的是整个数组的大小。

数组指针一般用于二维数组中。如:

void print_arr1(int arr[3][5], int row, int col)
{
    int i = 0;
    for (i = 0; i < row; i++)
    {
        for (j = 0; j < col; j++)
        {
            printf("%d ", arr[i][j]);
        }
        printf("\n");
    }
}
void print_arr2(int(*arr)[5], int row, int col)
{
    int i = 0;
    for (i = 0; i < row; i++)
    {
        for (j = 0; j < col; j++)
        {
            printf("%d ", arr[i][j]);
        }
        printf("\n");
    }
}
int main()
{
    int arr[3][5] = { 1,2,3,4,5,6,7,8,9,10 };
    print_arr1(arr, 3, 5);
    //数组名arr,表示首元素的地址
    //但是二维数组的首元素是二维数组的第一行
    //所以这里传递的arr,其实相当于第一行的地址,是一维数组的地址
    //可以数组指针来接收
    print_arr2(arr, 3, 5);
    return 0;
}

三、数组传参与指针传参

如何将【数组】或【指针】作为参数传给函数呢?

(一)数组传参

正确设计:

void test1(int arr[])
{}
void test1(int arr[10])
{}
void test1(int* arr)
{}
void test2(int* arr[20])
{}
void test2(int** arr)
{}
void test3(int arr[3][5])
{}
void test3(int arr[][5])
{}
void test3(int (*arr)[5])
{}
int main()
{
  int arr1[10] = { 0 };
  int* arr2[20] = { 0 };
  int arr3[3][5] = { 0 };
  test1(arr1);
  test2(arr2);
  test3(arr3);
}

错误设计:

void test3(int arr[][])//err
{}
void test3(int* arr)//err
{}
void test3(int* arr[5])//err
{}
void test3(int** arr)//err
{}
int main()
{
  int arr1[10] = { 0 };
  int* arr2[20] = { 0 };
  int arr3[3][5] = { 0 };
  test1(arr1);
  test2(arr2);
  test3(arr3);
}

错误分析:

首先要搞清楚函数参数的类型,test3(arr3)中明显传入的是arr3数组的首元素地址,又arr3为二维数组,所以该函数实际传入参数为arr3首行元素的地址,本质是指针

然后我们看函数定义中设计的参数:

void test3(int arr[][])二维数组可以省略行,但是不能省略列,所以该函数设计错误。

void test3(int* arr)此函数设计参数类型为指针数组,本质是数组,与传入参数不符,所以该函数设计错误。

void test3(int* arr[5])同上。

void test3(int** arr)此函数设计参数类型为二级指针数组,本质是数组,与传入参数不符,所以该函数设计错误。

(二)指针传参

void print(int* p, int sz)
{
  int i = 0;
  for (i = 0; i < sz; i++)
  {
    printf("%d\n", *(p + i));
  }
}
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9 };
  int* p = arr;
  int sz = sizeof(arr) / sizeof(arr[0]);
  //一级指针p,传给函数
  print(p, sz);
  return 0;
}

当函数的参数部分为一级指针时,该函数能接收什么参数呢?

比如:void test(char* p)

以下供参考:

void test(char* p)
{}
int main()
{
  char ch = 'w';
  char ptr = &ch;
  char arr[] = "abcdef";
  test(&ch);
  test(ptr);
  test(arr);
}

当函数的参数部分为二级指针时,该函数又能接收什么参数呢?比如:

void test1(int** p1)

void test2(char** p2)

以下供参考:

void test1(int** p)
{}
void test2(char** p)
{}
int main()
{
  int n = 10;
  int* p1 = &n;
  int** pp = &p1;
  test1(pp);
  test1(&p1);
  char c = 'b';
  char* pc = &c;
  char** ppc = &pc;
  char* arr[10];
  test2(&pc);
  test2(ppc);
  test2(arr);
  return 0;
}

今天的内容就分享到这,接下来博主还会继续跟进进阶指针的更新,关注博主不迷路🔥🔥🔥

目录
相关文章
|
2月前
|
存储 C语言
【C语言篇】深入理解指针3(附转移表源码)
【C语言篇】深入理解指针3(附转移表源码)
42 1
|
2月前
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
52 0
|
12天前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
2月前
|
C语言
【c语言】指针就该这么学(3)
本文介绍了C语言中的函数指针、typedef关键字及函数指针数组的概念与应用。首先讲解了函数指针的创建与使用,接着通过typedef简化复杂类型定义,最后探讨了函数指针数组及其在转移表中的应用,通过实例展示了如何利用这些特性实现更简洁高效的代码。
20 2
|
2月前
|
C语言
如何避免 C 语言中的野指针问题?
在C语言中,野指针是指向未知内存地址的指针,可能引发程序崩溃或数据损坏。避免野指针的方法包括:初始化指针为NULL、使用完毕后将指针置为NULL、检查指针是否为空以及合理管理动态分配的内存。
|
2月前
|
C语言
C语言:哪些情况下会出现野指针
C语言中,野指针是指指向未知地址的指针,通常由以下情况产生:1) 指针被声明但未初始化;2) 指针指向的内存已被释放或重新分配;3) 指针指向局部变量,而该变量已超出作用域。使用野指针可能导致程序崩溃或不可预测的行为。
|
2月前
|
存储 C语言
C语言32位或64位平台下指针的大小
在32位平台上,C语言中指针的大小通常为4字节;而在64位平台上,指针的大小通常为8字节。这反映了不同平台对内存地址空间的不同处理方式。
|
2月前
|
存储 算法 C语言
C语言:什么是指针数组,它有什么用
指针数组是C语言中一种特殊的数据结构,每个元素都是一个指针。它用于存储多个内存地址,方便对多个变量或数组进行操作,常用于字符串处理、动态内存分配等场景。
|
2月前
|
存储 C语言
C语言指针与指针变量的区别指针
指针是C语言中的重要概念,用于存储内存地址。指针变量是一种特殊的变量,用于存放其他变量的内存地址,通过指针可以间接访问和修改该变量的值。指针与指针变量的主要区别在于:指针是一个泛指的概念,而指针变量是具体的实现形式。
|
2月前
|
C语言
C语言指针(3)
C语言指针(3)
14 1