【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;
}

分析:

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