C语言之初识指针

简介: C语言之初识指针

一、认识指针


在认识指针之前,我们先要了解的是“内存”。


一般在购买电脑的时候,我们会很在意“内存”的大小,一般电脑的“内存”大小有4G、8G或者16G


他们之间的换算方式是:


1 TB = 1024 GB


1 GB =1024 MB


1 MB = 1024KB


1 KB = 1024B(字节)


1 B = 8 b(位)


程序在运行的时候会在“内存”中被调用,运行时占用“内存”的空间。


“内存”就好比一个大房子,里面有很多房间,为了能快速找到“内存”里面的值,我们给每个房间一个编号,这就是“内存编号”,这样就可以通过编号找到相应的值。


编号也被称为地址,不同“内存”大小的电脑地址编号表示不同,假如32位的电脑,它就会有32根地址线,那么假设每根地址线会产生高电平(高电压)和低电平(低电压)我们用1表是高电压,0表示低电压。


这样就产生了:


0000 0000 0000 0000 0000 0000 0000 0000~1111 1111 1111 1111 1111 1111 1111 1111种可能。


转化为16进制显示就是:0x00000000~0xFFFFFFFF(用二进制显示太多了,下图也是用16进制显示,不影响结果只是表现形式不同罢了)


那么指针是什么呢?


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


  1. 一般我们说的指针,通常指的是指针变量,是用来存放内存地址的变量



我们通常口头上说的指针是指指针变量,那么指针变量是什么呢?


指针变量也是一种变量,只不过是用来存储地址的变量。


我们可以通过&(取地址操作符)取出变量的“内存编号”,把地址可以存放到一个变量中,这个变量就是指针变量。


例如:


当我们定义一个整形变量a的时候。我们可以将a的地址存在一个指针变量p中,由于int占用四个字节,所以他有四个编号,电脑会存储其中最小的地址,假设首地址0X0000EF04.


#include <stdio.h>
int main()
{
  int a = 4;
  int* p = &a;//定义一个整形的指针,并指向变量a。
  printf("%p\n", &a);
  printf("%p", p);
  return 0;
}


图解:假设a的地址



实际在64位机器下a的地址:



指针大小:


在32位的电脑上,地址是32个0或者1组成二进制序列,那地址就得用32 b(位)=4 B(字节)的空间来存储,所以一个指针变量的大小就应该是4个字节。


那如果在64位机器上,如果有64个地址线,会有64个位,64 b=8 B(字节)的空间来存储地址

那一个指针变量的大小是8个字节,才能存放一个地址。


总结:


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


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


二、指针的类型


指针变量也是变量,我们在介绍变量的时候讲了变量有很多类型。那么指针变量也是如此。

当我们需要指向不同类型的变量时,就需要创建不同类型的指针变量。


注意:!!!


指针的大小与指针的类型无关,因为指针本质是用于存放被指向变量的地址,而地址的大小与计算机有关(32位或者64位…)


下面是在64位机器下,运行时观察不同类型的指针变量大小。


补充:指针是很危险的,因为它可以直接通过地址访问内存,我们在初始化指针的时候,当我们暂时还不明确要指向哪个变量的时候,我们可以将其初始化为空指针(NULL)。防止其访问非法的内存空间。


#include <stdio.h>
int main()
{
  char* p1 = NULL;//创建一个char类型的指针,并初始化为空指针
  int* p2 = NULL;//创建一个int类型的指针,并初始化为空指针
  short* p3 = NULL;
  long* p4 = NULL;
  float* p5 = NULL;
  double* p6 = NULL;
  //计算指针变量的大小
  printf("p1=%d,p2=%d, p3=%d, p4=%d, p5=%d, p6=%d", sizeof(p1), sizeof(p2), sizeof(p3), sizeof(p4), sizeof(p5), sizeof(p6));
  return 0;
}


运行结果:


p1=8,p2=8, p3=8, p4=8, p5=8, p6=8


2.1 指针类型的作用


从上面的代码我们可以知道,不同类型的指针大小相同,那么指针的类型还有什么意义呢?


还是代码容易解释:


🌰栗子


#include <stdio.h>
int main()
{
  int n = 10;
  char* p1 = (char*)&n;//将int类型地址强制转化为char类型
  int* p2 = &n;
  printf("n的地址是:  %p\n", &n);
  printf("p1:   %p\n", p1);
  printf("p1+1:   %p\n", p1 + 1);
  printf("p2:   %p\n", p2);
  printf("p2+1:   %p\n", p2 + 1);
  return 0;
}


运行结果:


n的地址是:      00000032FAAFFA24
p1:             00000032FAAFFA24
p1+1:           00000032FAAFFA25
p2:             00000032FAAFFA24
p2+1:           00000032FAAFFA28


图解:



所以不同类型的指针变量决定了指针向前或者向后走一步有多大(距离),即所谓步长。


2.2 指针的解引用


我们知道,指针保存被指向变量的地址,那么指针的作用是什么呢?


其实我们可以通过指针保存的地址,来访问内存中的目标变量,并将其修改。这就需要使用指针的解引用操作.


🌰栗子


补充知识: "%x"表示打印结果按照16进制打印,#表示打印前导符0X.


#include <stdio.h>
int main()
{
  int a = 0x11223344;
  int b = 0x11223344;
  int* p1 = &a;
  char* p2 = (char*)&b;
  //p1和p2都是指针,里面的内容是地址。
  //*p1是对指针解引用,即访问该地址所存储的变量。
  *p1 = 0;
  *p2 = 0;
  printf("%#x\n", a);
  printf("%#x", b);
  return 0;
}


运行结果:


0
0x11223300


结果分析:


因为p1是 int型指针变量,所以能访问四个字节的空间,p1 = 0,将a的值修改为0;


但是p2*是char类型指针变量 所以能访问一个字节的空间 ,所以只能修改b变量中 所以能访问一个字节的数据,至于为何是修改存储44这个字节的数据,就要涉及数据存储时字节的存储顺序知识了,在数据存储时,后面会讲到大小端的知识.现在我们只需要知道,char类型指针只能操作一个字节空间的数据就行了.


总结:


1.指针的类型决定指针加减整数时所跳过的步长.


2.指针的类型决定了,对指针解引用的时候有多大的权限(能操作几个字节)。


比如: char* 的指针解引用就只能访问一个字节,而 int* 的指针的解引用就能访问四个字节。


三、野指针


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


野指针是很危险的,它访问的空间不可知、


野指针出现的可能情况有:


  • 指针未初始化


  • 指针越界或者非法访问。


  • 指针指向的空间被释放了。(例如:局部变量)


🌰栗子(注意:下段代码是三种情况,此处是将他们写在了一个mian函数中.)


#include <stdio.h>
int main()
{
  //情况1:指针未初始化
  int* p1;//指针未初始化
  *p1 = 5;//在不知道指针指向的空间在哪时,强行改变指向空间的地址。
  //此时p1就是一个野指针。
  //情况2:指针越界访问
  int arr[5] = { 1,2,3,4,5 };
  int i = 0;
  int *p2 = arr;//p指向数组的首地址
  for (i = 0; i <= 10; i++)
  {
    printf("%d\n", *(p2++));
  }
  //当p2指向的空间超过数组的最后一个元素时,指针就非法访问,即越界访问了未知空间。
  //此时p2是野指针。
  //情况3:指向的空间被释放了
  int j = 0;
  for (j = 0; j < 5; j++)
  {
    int a = 5;
    a += 1;
  }
  int* p3 = &a;//当循环结束之后,局部变量a向内存申请的空间就被释放了。
  //此时p3指针指向的空间就是已经被释放掉的空间。p3就是野指针。
  return 0;
}


我们知道了野指针的危险,那么如何避免野指针的出现.


1. 指针初始化时明确指向的内容,不明确时设置为空指针.
2. 小心指针越界:使用指针是,仔细检查,指针是否会有越界的情况发生.
3. 指针指向空间释放,及时置NULL
4. 避免返回局部变量的地址
5. 指针使用之前检查有效性


四、指针的运算


4.1 指针加减整数


🌰栗子:一维数组的打印


#include <stdio.h>
int main()
{
  int arr[6] = { 1,2,3,4,5,6 };
  int sz = 0, i = 0;
  int* p = arr;//定义一个整形指针,指向arr数组的首元素
  sz = sizeof(arr) / sizeof(arr[0]);//计算元素的个数
  for (i = 0; i < sz; i++)
  {
    printf("%d ", *p + i);
  }
  return 0;
}


4.2 指针-指针


🌰栗子:自定义字符串长度计算函数


#include <stdio.h>
int my_strlen(const char* right)
{
  char *left = right;//保存数组的首元素地址
  while (*(right)!='\0')//指向'\0'
  {
    right++;
  }
  return right - left;//指向'\0'的指针-指向首元素的指针
}
int main()
{
  char c[] = { "primary-cattle still needs to study hard !" };
  int sz=my_strlen(c);
  printf("%d\n", sz);
  return 0;
}


图解:



指针-指针=元素之间的个数.


这里有一个细节:


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


五、二级指针


指针1是一种用来存放地址的变量,那么是变量也就有地址,所以指针也是有地址的.


当我们再创建一个指针2指向该指针1时 ,指针2就被称为2级指针.


#include <stdio.h>
int main()
{
  int a = 66;
  int* p1 = &a;
  int** p2 = &p1;//p2就是二级指针,指向p1
  **p2 = 4;
  printf("%d", a);
  return 0;
}


运行结果:


4


结果分析:


首先,int* p1 = &a;表示将,a的地址存放在指针p1中,


其次int** p2 = &p1;表示将p1的地址存放在p2指针中.


通过一次解引用找到p1,第二次解引用找到a,将其修改为4.


注意这两个’‘(星号)代表的意思,int * * p,第一个’‘(星号)代表被指向的对象是一个int类型,第二个’’(星号)表示p2是一个指针.


图解:



以此类推,还有三级指针和其他多级指针,常见的指针多为一级指针和二级指针,其他的几乎遇见不到.


好了,今天有关c语言中指针的基础知识就讲到这里了,相信大家对指针有所了解了,后续会将指针更深层的理解.

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