C语言之指针知识点总结

C语言之指针知识点总结

文章目录

  • C语言之指针知识点总结
    • 1. 初识指针
      • 1.1 取地址操作符 &
      • 1.2 指针变量
      • 1.3 解引用操作符 *
      • 1.4 指针变量
        • 1.4.1 大小
        • 1.4.2 指针类型的意义
      • 1.5 void*指针
      • 1.6 const关键字
        • 1.61 const修饰变量
        • 1.6.2 const修饰指针变量
      • 1.7 指针的运算
      • 1.7.1 指针+-整数
        • 1.7.2 指针 - 指针
        • 1.7.3 指针的关系运算
      • 1.8 野指针
        • 1.8.1 规避野指针
      • 1.9 指针的传值和传址
    • 2. 深入指针
      • 2.1 数组名的理解
      • 2.2 ⼀维数组传参的本质
      • 2.2 二级指针
      • 2.3 指针数组
      • 2.4 数组指针变量
      • 2.5 二维数组传参的本质
      • 2.6 函数指针变量
        • 2.6.1 函数指针变量的使用
        • 2.6.2 函数指针数组

1. 初识指针

1.1 取地址操作符 &

#include <stdio.h>int main()
{int a = 10;&a;return 0;
}

取地址操作符 & 用于取出变量的地址在这里插入代码片

1.2 指针变量

取出的地址要存放指针变量中

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

* 说明 p 是个指针变量,前面的int说明 指向的是一个int类型的对象

1.3 解引用操作符 *

解引用操作符是通过指针来找到指针指向的对象

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

*p 的意思就是通过pa中存放的地址,找到指向的空间,
p其实就是a变量了;所以p = 0,这个操作符是把a改成了20

1.4 指针变量

1.4.1 大小

指针变量的大小取决于系统,如果是32位平台下地址是32个bit位(即4个字节)
64位平台下地址是64个bit位(即8个字节)

1.4.2 指针类型的意义

一:指针的类型决定了,对指针解引⽤的时候有多⼤的权限(⼀次能操作⼏个字节)
二:指针的类型决定了指针向前或者向后⾛⼀步有多⼤(距离)

一:

#include <stdio.h>
int main()
{int m = 0x11223344;int n = 0x11223344;int* pi = &m;char* pc = &n;*pi = 0;*pc = 0;return 0;
}

上述代码中,pi会将m的4个字节全都改成0,而pc只会改n的1个字节改成0,由于指针类型不同,解引用时的权限不同

二:

#include <stdio.h>
int main()
{int n = 10;char *pc = (char*)&n;int *pi = &n;printf("%p\n", &n);printf("%p\n", pc);printf("%p\n", pc+1);printf("%p\n", pi);printf("%p\n", pi+1);return 0;
}

在这里插入图片描述
char* 的指针解引⽤就只能访问⼀个字节,⽽ int* 的指针的解引⽤就能访问四个字节

1.5 void*指针

void可以用来存放任意类型的地址,可以理解为无类型指针(或叫泛指型),但是void类型的指针不用解引用指针的±**

1.6 const关键字

1.61 const修饰变量
#include <stdio.h>int main()
{const n = 20;n = 10;//err 不可修改return 0;
}

当用const修饰变量时,被修饰的变量就无法更改它的值

1.6.2 const修饰指针变量
#include <stdio.h>int main()
{int a = 10;int b = 20;const int* x = &a; //const放在*左边 与下面一种写法一致int const* y = &a; //const放在*左边 int* const z = &a; //const放在*右边const int* const m = &a;//const放在*两边x = &b; *x = 10; /errz = &b;  /err*z = 20;m = &b;  /err*m = 20; /errreturn 0;
}

结论:
• const如果放在的左边,修饰的是指针指向的内容,保证指针指向的内容不能通过指针来改变。但是指针变量本⾝的内容可变。
• const如果放在
的右边,修饰的是指针变量本⾝,保证了指针变量的内容不能修改,但是指针指向的内容,可以通过指针改变

1.7 指针的运算

1.7.1 指针±整数

由于一维数组在内存中是连续存放的,那么就可以通过指针±整数,也就是指针偏移来遍历数组

代码一:

#include <stdio.h>
//指针+- 整数
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };int* p = &arr[0];int i = 0;int sz = sizeof(arr) / sizeof(arr[0]); //计算数组中的元素个数for (i = 0; i < sz; i++){printf("%d ", *(p + i));//p+i 这⾥就是指针+整数}return 0;
}

代码二:

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

代码一中的*(p + i) 就是 *(arr + i )
arr[ i ] 编译器会处理成 *(arr + i )
所以通过这两种方式都可以实现数组元素的打印

1.7.2 指针 - 指针
//指针-指针
#include <stdio.h>
int my_strlen(char* s)
{char* p = s;while (*p != '\0')p++;return p - s;
}
int main()
{printf("%d\n", my_strlen("abc"));return 0;
}

通过指针 - 指针的方式来模拟实现strlen
当*p中的值不为’\0‘时,如果指针加整数的方式,来找到’\0’的位置,然后通过高地址 - 低地址得出字符串的长度

指针 - 指针的绝对值是计算两个地址之间的元素个数

1.7.3 指针的关系运算
//指针的关系运算
#include <stdio.h>
int main()
{int arr[10] = {1,2,3,4,5,6,7,8,9,10};int *p = &arr[0];int i = 0;int sz = sizeof(arr)/sizeof(arr[0]); //计算数组中的元素个数while(p<arr+sz) //指针的⼤⼩⽐较{printf("%d ", *p);p++;}return 0;
}

p中存放的是数组中的首元素地址,
arr+sz 首元素地址加上数组元素个数,得到数组结尾的地址
当p的地址小于数组结尾的地址,打印*p
然后++找到下一个元素
就可以实现数组元素的打印

1.8 野指针

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

野指针的成因:

  1. 指针未初始化
  2. 指针越界访问
  3. 指针指向的空间释放

代码一:

#include <stdio.h>
int main()
{ int *p;//局部变量指针未初始化,默认为随机值*p = 20;return 0;
}

指针未初始化,不知道指向哪个空间,p是野指针

代码二:

#include <stdio.h>
int main()
{int arr[10] = {0};int *p = &arr[0];int i = 0;for(i=0; i<=11; i++){//当指针指向的范围超出数组arr的范围时,p就是野指针*(p++) = i;}return 0;
}

数组中只有10个元素,却循环了12次,指针访问越界了,p是野指针

代码三:

#include <stdio.h>
int* test()
{int n = 100;return &n;
}
int main()
{int*p = test();printf("%d\n", *p);return 0;
}

当进入test函数时,系统在内存中创建了一个空间用来存放n的值,当出test函数时,n的空间就被销毁了,所以p就指向了一个已经销毁的空间,p是野指针

1.8.1 规避野指针

1.初始化指针的时候,就给指针一个初始值,如果不知道指针要指向哪个空间,将指针置为NULL

#include <stdio.h>
int main()
{int num = 10;int*p1 = &num;int*p2 = NULL;return 0;
}

2.小心指针越界
内存有多大的空间,就让指针访问多大的空间,不能超出范围

3.避免返回局部变量的地址

1.9 指针的传值和传址

代码一:

#include <stdio.h>
void Swap(int x, int y)
{int tmp = x;x = y;y = tmp;
}
int main()
{int a = 0;int b = 0;scanf("%d %d", &a, &b);printf("交换前:a=%d b=%d\n", a, b);Swap1(a, b);printf("交换后:a=%d b=%d\n", a, b);return 0;
}

代码二:

#include <stdio.h>
void Swap(int* x, int* y)
{int tmp = *x;*x = *y;*y = tmp;
}
int main()
{int a = 0;int b = 0;scanf("%d %d", &a, &b);printf("交换前:a=%d b=%d\n", a, b);Swap1(&a, &b);printf("交换后:a=%d b=%d\n", a, b);return 0;
}

代码一:
无法交换两个变量的值,在传值调用中,改变形参的值无法改变形参的值
代码二:
传入的是两个变量的地址,在函数中改变形参就是改变实参,所以实现了两个变量的交换

2. 深入指针

2.1 数组名的理解

数组名就是数组首元素的地址
有两个情况例外:

sizeof(数组名) 计算的是整个数组的大小
&数组名 取出的是数组的地址,而非首元素的地址

2.2 ⼀维数组传参的本质

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

代码运行结果
sz1 = 10
sz2 = 1
一维数组传参传的是首元素的地址,所以无法通过函数来计算一维数组中的元素个数

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

结论:⼀维数组传参,形参的部分可以写成数组的形式,也可以写成指针的形式。

2.2 二级指针

指针变量也是变量,是变量就有地址,所以可以使用二级指针来存放指针变量的大小,同时也有三级指针,四级指针(三级指针之后就不常见了)

int main()
{int num = 10;int* p = &num;int** pp = &p;return 0;
}

在这里插入图片描述

2.3 指针数组

指针数组是存放指针的数组

int* p[4];

p是个指针数组,指向4个int*类型的元素

2.4 数组指针变量

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

int (*p)[5] = { 0 };

p先和*结合,说明p是⼀个指针变量变量,然后指着指向的是⼀个⼤⼩为10个整型的数组。所以p是⼀个指针,指向⼀个数组,叫 数组指针

[ ]的优先级要⾼于号的,所以必须加上()来保证p先和结合

2.5 二维数组传参的本质

#include <stdio.h>void test(int a[3][5], int r, int c)
{
int i = 0;
int j = 0;for (i = 0; i < r; i++){for (j = 0; j < c; j++){printf("%d ", a[i][j]);}printf("\n");}
}
int main()
{
int arr[3][5] = { {1,2,3,4,5}, {2,3,4,5,6},{3,4,5,6,7} };
test(arr, 3, 5);
return 0;
}

与一维数组不同的是,⼆维数组传参本质上也是传递了地址,传递的是第⼀
⾏这个⼀维数组的地址
与一维数组相同,这⾥实参是⼆维数组,形参也写成⼆维数组的形式,也可以写成指针的形式

总结:
一维数组的传参,传的是首元素的地址
二维数组的传参,传的是首元素的地址,不过二维数组的首元素是数组中第一行的元素
一维数组和二维数组的形参部分可以写成数组形式,也可以写成指针的形式

2.6 函数指针变量

函数也是有地址的,可以使用函数指针变量来存放函数的地址
与数组相同的是,函数名是函数的地址

int (*p)(int x,int y);
int (*p)(int, int);//x 和 y可以省略

其中int是函数的返回类型,p是函数指针变量名称,int int是p指向的函数的形参部分
去掉函数指针变量的名字,就是函数指针变量的类型
int (*)(int, int)

2.6.1 函数指针变量的使用
#include <stdio.h>
int Add(int x, int y)
{return x + y;
}
int main()
{int(*pf3)(int, int) = Add;printf("%d\n", (*pf3)(2, 3));printf("%d\n", pf3(3, 5));return 0;
}

通过函数指针变量来调用函数

2.6.2 函数指针数组

函数指针数组和指针数组一样,但是函数指针数组是用来存放函数指针变量的

int (*p[5])(int x, int y);
int (*p[5])(int, int);

p指针变量先与[ ] 结合,是数组,数组中 存放的是int (*)(int int)类型的函数指针

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/218658.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

定时器/计数器的应用

前言 对近期学习定时器进行简单的记录 参考链接 LED数码管的静态显示与动态显示&#xff08;KeilProteus&#xff09;-CSDN博客 外中断的应用-CSDN博客 【mcuclub】定时器/计数器_定时器/计数器的内部结构和工作方式-CSDN博客 5.图解定时器/计数器 - 知乎 (zhihu.com) 5…

C++ STL map迭代器失效问题

最近在开发过程中&#xff0c;定位一个问题的时候&#xff0c;发现多线程场景下大量创建和销毁某个C:\Windows\System32\reg.exe时出现了383个进程创建消息处理的接口&#xff0c;和384个进程销毁处理消息的接口都在等待锁&#xff0c;另外一个线程也在等锁&#xff0c;后面看了…

SQL sever2008中的游标

目录 一、游标概述 二、游标的实现 三、优缺点 3.1优点&#xff1a; 3.2缺点&#xff1a; 四、游标类型 4.1静态游标 4.2动态游标 4.3只进游标 4.4键集驱动游标 4.5显示游标&#xff1a; 4.6隐式游标 五、游标基本操作 5.1声明游标 5.1.1.IS0标准语法 5.1.1.1语…

【LLM_04】自然语言处理基础_2

一、神经网络1、循环神经网络&#xff08;RNN&#xff09;2、门控循环单元&#xff08;GRU&#xff09;3、长短期记忆网络&#xff08;LSTM&#xff09;4、双向RNN5、卷积神经网络&#xff08;CNN&#xff09; 二、注意力机制1、注意力机制原理介绍2、注意力机制的各种变式3、注…

Linux操作系统虚拟机安装(图文详解)

目录 前言 Linux系统介绍 虚拟机安装 1.安装步骤 2.破解激活步骤 3.创建Linux系统虚拟机 虚拟机的相关设置 1.基础设置 2.语言设置为中文 前言 今天我们开始学习Linux操作系统的安装虚拟机以及相关的Linux的环境配置&#xff0c;后面我还会继续发布Linux系统的相关基…

杰发科技AC7801——keil工程移植到IAR

0、简介 发现AC7801的代码只有keil工程的&#xff0c;IAR和Eclipse的代码只有一个例程&#xff0c;于是在从Keil移植到IAR时候遇到的问题记录下。 正常情况下&#xff0c;直接把keil的usr用户代码移植到iar的文件夹下面&#xff0c;删除原本的文件再添加新加进来的文件即可。…

深入浅出 Linux 中的 ARM IOMMU SMMU II

SMMU 驱动中的系统 I/O 设备探测 要使系统 I/O 设备的 DMA 内存访问能通过 IOMMU&#xff0c;需要将系统 I/O 设备和 IOMMU 设备绑定起来&#xff0c;也就是执行 SMMU 驱动中的系统 I/O 设备探测。总线发现系统 I/O 设备并和对应的驱动程序绑定&#xff0c;与 IOMMU 设备驱动程…

cephadm部署ceph quincy版本

环境说明 IP主机名角色 存储设备 192.168.2.100 master100 mon,mgr,osd,mds,rgw 大于5G的空设备192.168.2.101node101mon,mgr,osd,mds,rgw大于5G的空设备192.168.2.102node102mon,mgr,osd,mds,rgw大于5G的空设备 关闭防火墙 关闭并且禁用selinux 配置主机名/etc/hosts …

中伟视界:AI智能分析盒子实现全方位人车监测,保障管道安全

在油气管道长又无人的场景下&#xff0c;人和车的监测问题一直是一个难题。传统的监测手段往往存在盲区和误报问题&#xff0c;给管道运行安全带来了一定的隐患。然而&#xff0c;随着人工智能技术的不断发展&#xff0c;利用AI盒子的智能分析算法可以有效解决这一问题。 首先&…

Everything进行内网穿透搜索

文章目录 1\. 部署内网穿透1.1. 注册账号1.2. 登录1.3. 创建隧道 2\. 从外网访问Everything 借助cpolar可以让我们在公网上访问到本地的电脑 1. 部署内网穿透 1.1. 注册账号 在使用之前需要先进行注册cpolar cpolar secure introspectable tunnels to localhost 1.2. 登录 C…

cesium轨迹线(发光轨迹线)

cesium轨迹线(发光轨迹线) 下面有源码 实现思路 使用ellipse方法加载圆型,修改polyline中‘material’方法重写glsl来实现当前效果(cesium版本1.109) 示例代码 index.html <!DOCTYPE html> <html lang="en"><head

【LeetCode】挑战100天 Day16(热题+面试经典150题)

【LeetCode】挑战100天 Day16&#xff08;热题面试经典150题&#xff09; 一、LeetCode介绍二、LeetCode 热题 HOT 100-182.1 题目2.2 题解 三、面试经典 150 题-183.1 题目3.2 题解 一、LeetCode介绍 LeetCode是一个在线编程网站&#xff0c;提供各种算法和数据结构的题目&…