【C语言】 指针篇,不会的一定要看,指针没有想象的那么难,我们一起攻克它3

简介: 【C语言】 指针篇,不会的一定要看,指针没有想象的那么难,我们一起攻克它3

4.2 指针 - 指针

指针 - 指针 = 地址 - 地址

前提:

两个指针指向同一块空间(意味着指针的类型是一致的)

举例:

#include <stdio.h>
int main()
{
  int arr[10] = { 0,1,2,3,4,5,6,7,8,9 };
  int n = &arr[9] - &arr[0];
  printf("%d\n", n);
  return 0;
}

分析:


效果展示:

结论:指针 - 指针得到的是指针和指针之间的元素个数(大地址 - 小地址是正的,小地址 - 大地址是负的)

举例:求字符串长度

这个我们是用计数器的方法写的

#include <stdio.h>
int my_strlen(char* str)
{
  int count = 0;
  while (*str != '\0')
  {
    count++;
    str++;
  }
  return count;
}
int main()
{
  char str[] = "Hello World";
  int res = my_strlen(str);
  printf("%d\n", res);
  return 0;
}

我们接下来使用指针 - 指针的方式来写

#include <stdio.h>
int my_strlen(char* str)
{
  char* start = str;
  while (*str != '\0')
  {
    str++;
  }
  return str-start;
}
int main()
{
  char str[] = "Hello World";
  int res = my_strlen(str);
  printf("%d\n", res);
  return 0;
}

效果展示:

这里我们就是用到了结论。

4.3 指针的运算关系

我们来看一段代码:

#define N_VALUES 5
#include <stdio.h>
int main()
{
  float values[N_VALUES];
  float* vp;
  for (vp = &values[N_VALUES]; vp > &values[0];)
  {
    *--vp = 0;
  }
  return 0;
}

分析:


上面的代码可读性不高,我们对代码进行改进

#define N_VALUES 5
#include <stdio.h>
int main()
{
  float values[N_VALUES];
  float* vp;
  for (vp = &values[N_VALUES-1]; vp >= &values[0]; --vp)
  {
    *vp = 0;
  }
  return 0;
}

分析:


改进后的代码实际在绝大部分的编译器上是可以顺利完成任务的,然而我们还是应该避免这样写,因为标准并不保证它可行。


标准规定:


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


改进后的代码在循环判断部分是>=,这就存在了首元素的地址与首元素之前的地址进行比较,这样是不可以的。


画图深入理解一下:



5、指针和数组

指针就是指针;数组就是数组。不等同,但是有连系。

我们举例来深入了解:

#include <stdio.h>
int main()
{
  int arr[10] = { 0 };
  int* p = arr;//&arr[0] - 数组名是首元素地址
  int i = 0;
  for (i = 0; i < 10; i++)
  {
    *(p+i) = i + 1;
  }
  for (i = 0; i < 10; i++)
  {
    printf("%d ", *(p+i));
  }
  return 0;
}

效果展示:

Q:为什么可以使用指针对数组赋值呢?

A:

1.指针可以指向数组元素

2.因为指针可以运算,所以借助于指针可以访问数组


6、二级指针

我们来看这段代码:

#include <stdio.h>
int main()
{
  int a = 10;//a在内存中申请4个字节的空间
  int* pa = &a;//pa是指针变量,用来存放地址,也得向内存申请4/8个字节,因此pa也是有地址的
  int** ppa = &pa;//&pa的地址就放在ppa里面,因此ppa就是二级指针
  //打印a
  printf("%d\n", **ppa);
  return 0;
}

效果展示:


分析:




7、指针数组

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

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

数组我们已经知道整型数组、字符数组

我们以代码举例:

#include <stdio.h>
int main()
{
  int a = 10;
  int b = 20;
  int c = 30;
  //指针数组 - 存放指针数组
  int* arr[] = { &a, &b, &c };
  //打印
  int i = 0;
  for (i = 0; i < 3; i++)
  {
    printf("%d ", *(arr[i]));
  }
  return 0;
}

效果展示:


分析:

arr[ ]数组是 int* 类型的,说明存放的是指针,这就是指针数组,我们循环打印,对数组的每一位都解引用,就将数字全部打印出来了。

引申:

#include <stdio.h>
int main()
{
  char* ch[5];
  char** chh = ch;
  return 0;
}

分析:

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