首页 > 其他分享 >深入理解指针(三)

深入理解指针(三)

时间:2024-06-09 16:30:24浏览次数:28  
标签:arr NULL int 理解 深入 Swap main 指针

一、指针运算

1.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));
    }
    return 0;
}

我们可以看到p加上i,再对他们取地址,可以作为数组的下标,相减也同理。

1.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减去原长度s,所得到的就是字符串的长度。

1.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;
}

我们将for循环换成while循环,在里面将p和arr+sz进行比较,p<arr+sz其实就是1.1中的i<sz。

二、野指针

概念:野指针就是至臻纸箱的位置时不可知的(随机的、不正确的)

2.1野指针的成因

2.1.1.指针未初始化

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

2.1.2指针越界访问

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

2.1.3指针指向的空间释放

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

2.2如何规避野指针

2.2.1指针初始化

如果明确知道指针指向哪里就直接赋值,如果不知道,可以给指针赋值NULL。NULL是C语言中定义的一个标识符常亮,值是0,0也是地址,这个地址是无法使用的,读写改地址会报错。

初始化如下:

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

2.2.2小心指针越界

一个程序向内存申请了哪些空间,就只能访问哪些,不能超出范围去访问,超出了就是越界访问。

2.2.3指针变量不再使用时,及时置NULL,指针使用之前检查有效性。

        当指针变量指向⼀块区域的时候,我们可以通过指针访问该区域,后期不再使用这个指针访问空间的时候,我们可以把该指针置为NULL。因为约定俗成的⼀个规则就是:只要是NULL指针就不去访问,同时使用指针之前可以判断指针是否为NULL。
        我们可以把野指针想象成野狗,野狗放任不管是非常危险的,所以我们可以找⼀棵树把野狗拴起来,就相对安全了,给指针变量及时赋值为NULL,其实就类似把野狗栓起来,就是把野指针暂时管理起来。不过野狗即使拴起来我们也要绕着走,不能去挑逗野狗,有点危险;对于指针也是,在使用之前,我们也要判断是否为NULL,看看是不是被拴起来起来的野狗,如果是不能直接使用,如果不是我们再去使用。

#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;
    for (i = 0; i < 10; i++)
    {
        *(p++) = i;
    }
    //此时p已经越界了,可以把p置为NULL
    p = NULL;
    //下次使用的时候,判断p不为NULL的时候再使用
    p = &arr[0];//重新让p获得地址
    if (p != NULL)//判断
    {
        //...
    }
    return 0;
}

2.2.4避免返回局部变量的地址

2.1.3中的例子,不要返回局部变量的地址。

三、assert断言

        assert.h这个头文件中定义了宏assert(),用于在运行时确保符合指定条件,如果不符合就报错终止运行。这个宏常常被称为“断言”。

assert(p != NULL);

        在程序运行到上述代码时,如果p不等于NULL,程序继续运行,如果等于,就会终止运行,并且给出报错信息提示。

        assert() 宏接受一个表达式作为参数。如果该表达式为真(返回值非零), assert() 不会产生
任何作用,程序继续运行。如果该表达式为假(返回值为零), assert() 就会报错,在标准错误
流 stderr 中写入一条错误信息,显示没有通过的表达式,以及包含这个表达式的文件名和行号。
        assert() 的使用对程序员是非常友好的,使用 assert() 有几个好处:它不仅能自动标识文件和
出问题的行号,还有一种无需更改代码就能开启或关闭 assert() 的机制。如果已经确认程序没有问
题,不需要再做断言,就在 #include <assert.h> 语句的前面,定义⼀个宏 NDEBUG 。

#define NDBUG

#include<assert.h>

        然后,重新编译程序,编译器就会禁用文件中所有的 assert() 语句。如果程序又出现问题,可以移除这条 #define NDEBUG 指令(或者把它注释掉),再次编译,这样就重新启用了 assert() 语
句。
        assert() 的缺点是,因为引入了额外的检查,增加了程序的运行时间。一般我们可以在 Debug 中使用在 Release 版本中选择禁用 assert 就行,在 VS 这样的集成开发环境中,在 Release 版本中,直接就是优化掉了。这样在debug版本写有利于程序员排查问题,在 Release 版本不影响用户使用时程序的效率。

四、指针的使用和传址调用

例:写出一个函数,交换两个整型变量的值

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

代码运行的结果如下:

这是为什么呢?

        我们发现在main函数内部,创建了a和b,a的地址是0x0093fe58,b的地址是0x0093fe4c,在调用Swap函数时,将a和b传递给了Swap函数,在Swap函数内部创建了形参x和y接收a和b的值,但是x的地址是0x0093fd74,y的地址是0x0093fd78,x和y确实接收到了a和b的值,不过x的地址和a的地址不一样,y的地址和b的地址不一样,相当于x和y是独立的空间,那么在Swap函数内部交换x和y的值,自然不会影响a和b,当Swap函数调用结束后回到main函数,a和b的没法交换。Swap函数在使用的时候,是把变量本身直接传递给了函数,这种调用函数的方式我们之前在函数的时候就知道了,这种叫传值调用。

结论:实参传递给形参的时候,形参会单独创建一份临时空间来接收实参,对形参的修改不影响实参。

那怎么办呢?

        我们现在要解决的就是当调用Swap函数的时候,Swap函数内部操作的就是main函数中的a和b,直接将a和b的值交换了。那么就可以使用指针了,在main函数中将a和b的地址传递给Swap函数,Swap函数里边通过地址间接的操作main函数中的a和b,并达到交换的效果就好了。

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

我们可以看到实现成Swap的方式,顺利完成了任务,这里调用Swap函数的时候是将变量的地址传
递给了函数,这种函数调用方式叫:传址调用。

标签:arr,NULL,int,理解,深入,Swap,main,指针
From: https://blog.csdn.net/mingxunwwww/article/details/139546844

相关文章

  • 深入浅出Rust所有权:手把手从零设计Rust所有权体系,掌握Rust内存管理思想的精髓
    撰写编程语言发展历史过程中,对Rust的所有权机制的设计进行了深入的探讨,摘取其中的一段内容,邀请大家点评。Rust的所有权机制,看似复杂且与现有编程语言不同,使用起来思路也许难以适应。是学习Rust的难点。但如果我们换个思路,假设我们是Rust的设计者,逐步深入Rust的内心世界,也许......
  • 深入了解Git:从数据模型到集成IDEA
    Git是现代软件开发中不可或缺的版本控制工具。理解Git的数据模型、暂存区、命令行接口,并将其集成到IDE(如IntelliJIDEA),可以显著提升开发效率。本文将从底层开始,逐步深入Git的各个方面,并介绍如何将其集成到IntelliJIDEA中。目录Git的数据模型暂存区Git的命令行接口将Git集......
  • 深入解析C++中自动生成默认构造函数的五种情况
    自动生成默认构造函数的情况以及相关解释在C++中,当一个类没有任何用户定义的构造函数时,编译器会自动为这个类生成一个默认构造函数。以下是具体情况的解释以及示例:1.带有默认构造函数的类成员对象如果一个类没有任何构造函数,但它含有一个成员对象,而该成员对象有默认构造......
  • 深入理解交叉熵损失 CrossEntropyLoss - CrossEntropyLoss
    深入理解交叉熵损失CrossEntropyLoss-CrossEntropyLossflyfish本系列的主要内容是在2017年所写,GPT使用了交叉熵损失函数,所以就温故而知新,文中代码又用新版的PyTorch写了一遍,在看交叉熵损失函数遇到问题时,可先看链接提供的基础知识,可以有更深的理解。深入理解交叉熵损......
  • 【源码】Spring Data JPA原理解析之事务执行原理
     SpringDataJPA系列1、SpringBoot集成JPA及基本使用2、SpringDataJPACriteria查询、部分字段查询3、SpringDataJPA数据批量插入、批量更新真的用对了吗4、SpringDataJPA的一对一、LazyInitializationException异常、一对多、多对多操作5、SpringDataJPA自定义......
  • 深入浅出,解析ChatGPT背后的工作原理
    自ChatGPT发布以来,已经吸引了无数人一探究竟。但ChatGPT实际上是如何工作的?尽管它内部实现的细节尚未公布,我们却可以从最近的研究中一窥它的基本原理。ChatGPT是OpenAI发布的最新语言模型,比其前身GPT-3有显著提升。与许多大型语言模型类似,ChatGPT能以不同样式、不......
  • 深入解析:MySQL连接超时问题排查与优化策略
    引言​在现代企业应用中,数据库的稳定性和响应速度是保证业务连续性的关键。MySQL作为广泛使用的数据库系统,其连接超时问题可能成为性能瓶颈,影响用户体验和业务效率。本文将深入探讨MySQL连接超时的原因、影响以及优化策略。超时配置详解​查看当前设置​要了解MySQL......
  • C语言中的指针(1)
    目录指针是什么?指针变量&和*操作符指针变量的类型指针变量的大小指针变量类型的意义void*指针const修饰指针const修饰变量const修饰指针变量指针运算•指针+-整数•指针-指针​编辑​编辑​编辑•指针的关系运算野指针概念成因指针未初始化​编......
  • 深入解析Kafka消息丢失的原因与解决方案
    深入解析Kafka消息丢失的原因与解决方案ApacheKafka是一种高吞吐量、分布式的消息系统,广泛应用于实时数据流处理。然而,在某些情况下,Kafka可能会出现消息丢失的情况,这对于数据敏感的应用来说是不可接受的。本文将深入解析Kafka消息丢失的各种原因,包括生产者、broker和消费......
  • 深入解析Kafka消息传递的可靠性保证机制
    深入解析Kafka消息传递的可靠性保证机制Kafka在设计上提供了不同层次的消息传递保证,包括atmostonce(至多一次)、atleastonce(至少一次)和exactlyonce(精确一次)。每种保证通过不同的机制实现,下面详细介绍Kafka如何实现这些消息传递保证。1.AtMostOnce(至多一次)在这种模......