C语言初阶指针详解

简介: 目录1. 指针是什么2. 指针和指针类型3. 野指针4. 指针运算5. 指针和数组6. 二级指针7. 指针数组

1. 指针是什么

指针理解的2个要点:

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

2. 平时口语中说的指针,通常指的是指针变量,是用来存放内存地址的变量

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

#include<stdio.h>
int main()
{
  int a = 10;
  int* pa = &a;  //pa是指针变量
  printf("%d\n", sizeof(pa));  // 4
  //  printf("%p\n", &a);
  //  00000000 00000000 00000000 00001010
  //  00  00  00  0a
  return 0;
}

image.png

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


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


那如果在64位机器上,如果有64个地址线,那一个指针变量的大小是8个字节,才能存放一个地 址。


总结: 指针是用来存放地址的,地址是唯一标示一块地址空间的。 指针的大小在32位平台是4个字节,在64位平台是8个字节。


2. 指针和指针类型

#include<stdio.h>
int main()
{
  char* pc;
  int* pa;
  double* pd;
  printf("%d\n",sizeof(pc));    // 4
  printf("%d\n", sizeof(pa));   // 4
  printf("%d\n", sizeof(pd));   // 4
  return 0;
}

指针作用1:指针类型决定了在解引用时能访问几个字节(指针的权限)

#include<stdio.h>
int main()
{
  int a = 0x11223344;
  int* pa = &a;
  *pa = 0;
  char* pc = &a;
  *pc = 0;
  // 指针类型决定了在解引用的时候以此能访问几个字节(指针的权限)
  // int*     --> 4
  // char*    --> 1
  // double*  --> 8
  return 0;
}

指针作用2:指针类型决定了//指针作用2:指针类型决定了指针向前或者向后走一步,走多大距离(单位是字节)

#include<stdio.h>
int main()
{
  int a = 10;
  int* pa = &a;
  char* pc = &a;
  printf("%p\n", pa);     //001EFE54
  printf("%p\n", pa + 1); //001EFE58
  printf("%p\n", pc);     //001EFE54
  printf("%p\n", pc + 1); //001EFE55
  return 0;
}

例如:

创建一个整型数组,10个元素

1:初始化数组的内容是1-10

2:打印数组

1.
#include<stdio.h>
int main()
{
  int arr[10] = { 0 };
  int* p = arr;
  int i = 0;
  for (i = 0; i < 10; i++)
  {
    *(p + i) = i+1;
  }
  // 倒着打印
  int* q = &arr[9];
  for (i = 0; i < 10; i++)
  {
    printf("%d ", *q);  // 10 9 8 7 6 5 4 3 2 1 
    q--;
  }
  return 0;
}

3. 野指针

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

成因:

#include<stdio.h>
int main()
{
  /*int a;
  printf("%d\n", a);*/
  //未初始化
  //int* p;  //野指针
  //*p = 20;
  //越界访问
  int arr[10] = { 0 };
  int* p = arr;
  int i = 0;
  for (i = 0; i <= 10; i++)
  {
    *p = i;
    p++;
  }
  return 0;
}

如何规避野指针?如下:

1. 指针初始化

2. 小心指针越界

3. 指针指向空间释放即使置NULL

4. 避免返回局部变量的地址 5. 指针使用之前检查有效性

#include<stdio.h>
int* test()
{
  int a = 100;
  return &a;
}
int main()
{
  int* p = test();
  printf("%d\n", *p);  // 100
  return 0;
}
// 指针指向空间及时放置NULL
#include<stdio.h>
int main()
{
  int* p = NULL;
  if (p != NULL)
  {
    *p = 100;
  }
  int arr[10] = { 0 };
  int* q = arr;
  q = NULL;
  return 0;
}

4. 指针运算

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

倒着打印:

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

指针减指针

#include<stdio.h>
int main()
{
  int a[10] = { 0 };
  printf("%d\n", &a[9] - &a[0]);    //   9
  printf("%d\n", &a[0] - &a[9]);    //  -9
  return 0;
}
//相减的前提是两块指针指向同一块空间
#include<stdio.h>
int main()
{
  int a = 10;
  char c = 'w';  
  &a - &c;   //  编译器报错,不能不同类型指针相加减
  return 0;
}

求字符串长度(拓展)指针-指针

#include<stdio.h>
#include<string.h>
int my_strlen(char* s)
{
  char* start = s;
  while (*s != '\0')
  {
    s++;
  }
  return s - start;
}
int main()
{
  char arr[] = "abc";
  int len = my_strlen(arr);
  printf("%d\n", len);     // 3
  return 0;
}

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;
  int sz = sizeof(arr) / sizeof(arr[0]);
  /*for (i = 0; i < 10; i++)
  {
    printf("%p==%p\n", p + i, &arr[i]);
  }*/
  for (i = 0; i < 10; i++)
  {
    printf("%d ", p[i]);    // 1 2 3 4 5 6 7 8 9 10
  }
  return 0;
}

image.png

6. 二级指针

指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪里? 这就是 二级指针image.png

#include<stdio.h>
int main()
{
  int a = 10;
  int* pa = &a;
  int** ppa = &pa;  //ppa就是一个二级指针
  int*** pppa = &ppa;//pppa就是一个三级指针
  //***pppa = 20;
  *pa = 20;
  printf("%d\n", a);
  return 0;
}

image.png

7. 指针数组

#include<stdio.h>
int main()
{
  int arr[10]; //整型数组,存放整形的数组
  char ch[5];  //字符数组,存放字符的数组
  //指针数组,存放指针的数组
  int a = 10;
  int b = 20;
  int c = 30;
  int* arr2[5] = { &a,&b,&c }; //存放整形指针的数组
  int i = 0;
  for (i = 0; i < 5; i++)
  {
    printf("%d ", *(arr2[i]));   // 10 20 30
  }
  return 0;
}

指针数组是指针还是数组?

答案:是数组。是存放指针的数组。

数组我们已经知道整形数组,字符数组。

image.png


那指针数组是怎样的?

image.png

arr3是一个数组,有五个元素,每个元素是一个整形指针。

image.png

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