c语言分层理解(c语言指针(上))

简介: 1.指针是什么?指针是内存中的一个最小单元的编号,也就是地址。

1.指针是什么?

指针是内存中的一个最小单元的编号,也就是地址。

怎么来理解这句话呢?看图:

c4250426c4a8a86005cc0f8ea0f13765.png

平时我们口语中所说的指针通常是指针变量,是用来存放内存地址的变量。

总结:指针就是地址,口语中所说的指针通常是指针变量,用来存放地址。

初步了解指针概念后再来看看这个图:

0d5f97a2bedc3bbe80080870c8db99b8.png

这里就更清楚内存的概念。

再来说说指针变量:

我么通过&操作符取出变量的内存起始地址,把地址可以存放在一个变量中,这个变量就是指针变量。而指针变量也是有地址的。

bcbd76c479c4958dee8f60f3ac68d4e2.png

22993d9906afd61f3e359397d9d09d9f.png

总结:

指针变量是用来存放地址的变量。

这里有两个问题:1.一个小的单元到底是多大?2.如何编址的?

经过仔细的计算和权衡发现一个字节给一个对应的地址是比较合适的。

对于32位机器,假设由32跟地址线,那么假设每跟地址线在寻址的时候产生高电压和低电压也就是1或者0.那么32跟地址线产生的地址会是:

00000000 00000000 00000000 00000000

00000000 00000000 00000000 00000001

11111111 11111111 11111111 11111111

这里就是232次方个地址。

每个地址标识一个字节,那么我们就给(232byte == 232/1024kb == 232/1024/1024mb=232/1024/1024/1024 == 4GB)4G的空间进行编址,那么64位机器下就是232*4GB的空间。


恍然大悟:在32位机器上,地址是32个0或者1组成的二进制序列,那地址得用4个字节的空间来存储,所以指针变量的大小是4个字节。在64位机器上,如果有64跟地址线,那么一个指针变量的大小就是8个字节。

再看一遍:


faed53dfe79bbe4c287609ea6e2a7562.png

总结:

指针变量是用来存放地址的,地址是唯一标识一个内存单元的。

指针的大小在32位平台上是4个字节,在64位平台上是8个字节。

2.指针和指针类型

指针有很多类型,比如:

int*、double*、float*、char*、short* 等等

我们说既然指针在内存中相同平台下(32位平台或者64位平台)都是一样的大小,那么我们要指针的类型有什么意义呢?

通过例子来说明指针类型有什么用。

2.1 指针解引用

e6856b3b0c2c40d3abc8dac8ed4b8e74.png

2.2 指针±整数

image.png

总结:指针类型其实是决定解引用时访问几个字节的大小以及决定了指针的步长

3.野指针

概念: 野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)

3.1 野指针成因

3.1.1 指针未初始化


e8a36b0c33d3cc6af771ecfc2897d031.png

3.1.2 指着越界访问

#include <stdio.h>
int main()
{
  int arr[10] = { 0 };
  int i = 0;
  int sz = sizeof(arr) / sizeof(arr[0]);
  int* p = arr;
  for (i = 0; i <= sz; i++)//这里当下标为sz时就数组越界访问了
  {
    *p = i;
    p++;
  }
  return 0;
}

3.1.3 指针指向的空间释放

动态内存管理中会着重讲解,这里点一下。

int* test()
{
  int num = 100;
  return &num;
}
int main()
{
  int* p = test();
  *p = 200;
  return 0;
}

这段代码看上去其实没什么问题,问题就是看上没什么问题,其实问题在于函数调用时开辟内存空间,结束时释放内存空间。所以这里的num在函数执行完后,又返回给了操作系统,这里的&num不知道取的哪里的地址,很危险,再解引用对其这一个块内存空间修改,这就是一个野指针问题。

3.2 如何规避野指针

1.指针初始化

2.小心指针越界

3.指着指向空间释放,及时置为NULL(空指针)

4.避免返回局部变量地址

5.指针使用之前要检查是否具有有效性

对第5点进行解释:

#include <stdio.h>
int main()
{
  int* p = NULL;
  int a = 20;
  p = &a;
  if (p != NULL)
  {
    *p = 20;
  }
  return 0;
}

如果不检测,假如给一个空指针的指针变量对其给改值:


60a228e4dd875635e47d64b236937291.png

4.指针运算

4.1 指针±整数

#include <stdio.h>
#define N_VALUES 5
float values[N_VALUES];
float* vp;
//指针+-整数;指针的关系运算
int main()
{
  for (vp = &values[0]; vp < &values[N_VALUES];)
  {
    *vp++ = 0;
  }
  return 0;
}

分析:

ad4bca317d001d3fc45c81ffcabe1566.png

4.2 指针-指针

指针-指针是有意义的,使用场景是求数组元素个数等等。

在生活中,日期-日期=天数,但是你要是日期+日期这有什么意义呢,指针也是一样,指针+指针没有使用场景。

举一个例子:

//计算字符串中元素个数
#include <stdio.h>
int my_strlen(char *p)
{
  char* ptr = p;
  while (*p != '\0')
  {
    p++;
  }
  return p-ptr;
}
int main()
{
  char arr[] = "abcdef";
  int num = my_strlen(arr);
  printf("%d\n", num);
  return 0;
}

指针的关系运算

a6a0e0cedbce0fcde737b220fee48f0f.png

两段代码相比一个是访问数组前的地址,一个是访问后的地址。这实际上是可以顺利完成任务的,但是应该避免上面第一种写法,因为标准规定:

允许指向数组元素的指针与指向数组最后一个元素后面的那个内存位置的指针比较,但是不允许与指向第一个元素之前的那个内存位置的指针进行比较。

5.指针和数组

我们知道了数组名就是首元素地址,那么我们可以通过指针来间接访问数组元素,如:

#include <stdio.h>
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
  int* p = arr;
  int i = 0;
  for (i = 0; i < 10; i++)
  {
    printf("%d ", *(p + i));
  }
  return 0;
}//最终输出1 2 3 4 5 6 7 8 9 10

因为数组在内存中的地址是连续存放的,所以可以用指针来访问每个元素的地址。

数组这方面知识比较薄弱可以点这个链接来看看哟c语言数组!

6.二级指针

通过一个实例,来观察二级指针:

1c563020becc95314c52bb0ba8de938f.png

这里怎么去解释呢?


812ce0431f9480a0419969ef013e6482.png

这里有二级指针的概念那么有三级指针或者四级指针吗?理论上是有的,下面来看看实例:

fa3a94fa09b82fffc5638bdf0f73d7ef.png

这里和二级指针逻辑一样,就不再说明关系了。

7.指针数组

首先我们要搞明白指针数组是什么?

其实指针数组是一个存放指针的数组。

还是用实例来理解:

#include <stdio.h>
int main()
{
  int a = 10;
  int b = 20;
  int c = 30;
  int d = 40;
  int e = 50;
  int* arr[5] = { &a,&b,&c,&d,&e };
  int i = 0;
  for (i = 0; i < 5; i++)
  {
    printf("%d ", *(arr[i]));
  }
  printf("\n");
  for (i = 0; i < 5; i++)
  {
    printf("%p\n", arr[i]);
  }
  printf("\n");
  printf("%p\n", &a);
  printf("%p\n", &b);
  printf("%p\n", &c);
  printf("%p\n", &d);
  printf("%p\n", &e);
  return 0;
}

通过代码进行分析:

b436c3c34127e839610a5e71581950d0.png

再看内存地址:

9bc60838b794782bcbaef8c5bfd57bb9.png

分析得到:

数组中的元素在内存是连续存放的,但是指针数组中的元素在内存中是随机存放的,因为变量是创建后,地址拿到数组中存放,千万别把数组的这个性质也和指针数组搞的一样,那就有点尴尬了。

再看一个应用场景:

#include <stdio.h>
//用一维数组模拟一个二维数组
//前提是必须每个一维数组的元素个数一样
int main()
{
  int arr1[] = { 1,2,3,4,5 };
  int arr2[] = { 2,3,4,5,6 };
  int arr3[] = { 3,4,5,6,7 };
  int arr4[] = { 4,5,6,7,8 };
  //指针数组arr中存的是每个数组的数组名(也就是每个数组的首元素地址)
  int* arr[] = { arr1,arr2,arr3,arr4 };
  //外层for循环指指针数组arr
  int i = 0;
  for (i = 0; i < 4; i++)
  {
    //内存for循环指指针数组中的内容
    int j = 0;
    for (j = 0; j < 5; j++)
    {
      printf("%d ", arr[i][j]);
    }
    printf("\n");
  }
  return 0;
}
#include <stdio.h>
int main()
{
  int arr1[] = { 1,2,3,4,5 };
  int arr2[] = { 2,3,4,5,6 };
  int arr3[] = { 3,4,5,6,7 };
  int arr4[] = { 4,5,6,7,8 };
  int* arr[] = { arr1,arr2,arr3,arr4 };
  int i = 0;
  for (i = 0; i < 4; i++)
  {
    int j = 0;
    for (j = 0; j < 5; j++)
    {
      printf("%d ", *( * (arr + i) + j));
    }
    printf("\n");
  }
  return 0;
}

逻辑上不用分析,这里我们来看看指针数组的存储以及两种打印形式的理解:


c613ad49de1f8f43415afaec1ed5d97a.png








相关文章
|
1月前
|
存储 NoSQL 编译器
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
指针是一个变量,它存储另一个变量的内存地址。换句话说,指针“指向”存储在内存中的某个数据。
97 3
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
|
1月前
|
存储 编译器 C语言
【C语言】指针大小知多少 ?一场探寻C语言深处的冒险 !
在C语言中,指针的大小(即指针变量占用的内存大小)是由计算机的体系结构(例如32位还是64位)和编译器决定的。
62 9
|
1月前
|
安全 程序员 C语言
【C语言】指针的爱恨纠葛:常量指针vs指向常量的指针
在C语言中,“常量指针”和“指向常量的指针”是两个重要的指针概念。它们在控制指针的行为和数据的可修改性方面发挥着关键作用。理解这两个概念有助于编写更安全、有效的代码。本文将深入探讨这两个概念,包括定义、语法、实际应用、复杂示例、最佳实践以及常见问题。
49 7
|
2月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
191 13
|
2月前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
2月前
|
算法 C语言
C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项
本文深入讲解了C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项,通过实例演示了文件操作的基本流程,帮助读者掌握这一重要技能,提升程序开发能力。
153 3
|
2月前
|
存储 算法 程序员
C 语言指针详解 —— 内存操控的魔法棒
《C 语言指针详解》深入浅出地讲解了指针的概念、使用方法及其在内存操作中的重要作用,被誉为程序员手中的“内存操控魔法棒”。本书适合C语言初学者及希望深化理解指针机制的开发者阅读。
|
2月前
|
程序员 C语言
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门。本文深入探讨了指针的基本概念、声明方式、动态内存分配、函数参数传递、指针运算及与数组和函数的关系,强调了正确使用指针的重要性,并鼓励读者通过实践掌握这一关键技能。
47 1
|
2月前
|
存储 C语言 计算机视觉
在C语言中指针数组和数组指针在动态内存分配中的应用
在C语言中,指针数组和数组指针均可用于动态内存分配。指针数组是数组的每个元素都是指针,可用于指向多个动态分配的内存块;数组指针则指向一个数组,可动态分配和管理大型数据结构。两者结合使用,灵活高效地管理内存。
|
2月前
|
存储 NoSQL 编译器
C 语言中指针数组与数组指针的辨析与应用
在C语言中,指针数组和数组指针是两个容易混淆但用途不同的概念。指针数组是一个数组,其元素是指针类型;而数组指针是指向数组的指针。两者在声明、使用及内存布局上各有特点,正确理解它们有助于更高效地编程。