热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

引用超出范围的char*

如何解决《引用超出范围的char*》经验,为你挑选了4个好方法。

我最近在用C++编程了一段时间之后再次开始用C编程,而我对指针的理解有点生疏.

我想问一下为什么这段代码没有导致任何错误:

char* a = NULL;
{
    char* b = "stackoverflow";
    a = b;
}

puts(a);

我认为因为b超出范围,a应该引用一个不存在的内存位置,因此它们在调用时会出现运行时错误printf.

我在MSVC中运行此代码大约20次,并且没有显示错误.



1> dbush..:

b定义的范围内,为其分配字符串文字的地址.这些文字通常位于内存的只读部分,而不是堆栈.

当你做a=b你指定ba,即a现在包含一个字符串的地址.b超出范围后,此地址仍然有效.

如果你采取了地址b,然后试图取消引用该地址,那么你就调用未定义行为.

所以,你的代码是有效的,并没有发生未定义行为,但下列情况:

int *a = NULL;
{
    int b = 6;
    a = &b;
}

printf("b=%d\n", *a);

另一个更微妙的例子:

char *a = NULL;
{
    char b[] = "stackoverflow";
    a = b;
}

printf(a);

此示例与您的示例之间的区别在于b,它是一个数组,在指定时会衰减到指向第一个元素的指针a.所以在这种情况下a包含一个局部变量的地址,然后超出范围.

编辑:

作为旁注,将变量作为第一个参数传递是不好的做法printf,因为这会导致格式字符串漏洞.最好使用字符串常量如下:

printf("%s", a);

或者更简单:

puts(a);


我认为即使后一个代码也可能优化为`printf("stackoverflow")`取决于你正在使用的编译器/开关
@CometEngine很高兴我能提供帮助.如果您发现它有用,请随意[接受此答案](/sf/ask/17360801/).
@Govind Parmar刚刚在msvc上进行了全面优化和内联测试; 似乎是这样的.

2> SiggiSv..:

逐行,这是您的代码所做的:

char* a = NULL;

a是一个不引用任何东西的指针(设置为NULL).

{
    char* b = "stackoverflow";

b是一个引用静态常量字符串文字的指针"stackoverflow".

    a = b;

a设置为也引用静态,常量字符串文字"stackoverflow".

}

b超出范围.但是,由于a没有引用b,那么这也不要紧(它只是引用相同的静态常量字符串文字作为b被引用).

printf(a);

打印"stackoverflow"引用的静态常量字符串文字a.


空指针不"指向NULL".它*是*NULL.

3> kyle..:

字符串文字是静态分配的,因此指针无限期有效.如果您已经说过char b[] = "stackoverflow",那么您将在堆栈上分配一个char数组,当该范围结束时该数组将变为无效.这种差异也表现为修改字符串:char s[] = "foo"stack分配一个你可以修改的字符串,而char *s = "foo"只给你一个指向可以放在只读内存中的字符串的指针,所以修改它是未定义的行为.



4> zwol..:

其他人已经解释说这段代码完全有效.这个答案是关于你的期望,如果代码无效,调用时会出现运行时错误printf.不一定如此.

让我们看一下代码中的这种变化,这无效的:

#include 
int main(void)
{
    int *a;
    {
        int b = 42;
        a = &b;
    }
    printf("%d\n", *a); // undefined behavior
    return 0;
}

这个程序有不确定的行为,但它恰好是相当可能的,这将在事实上,印刷品42几个不同的原因-许多编译器将离开堆栈槽b分配的整个身体main,因为没有别的需要的空间和最小化堆栈调整的数量简化了代码生成; 即使编译没有正式解除分配栈槽,数字42很可能保留在内存中,直到别的东西覆盖它,并没有什么之间a = &b,并*a做到这一点; 标准优化("常量和复制传播")可以消除这两个变量,并将最后已知的值*a直接写入printf语句中(就像您已经编写过一样printf("%d\n", 42)).

理解"未定义的行为"并不意味着"程序会以可预测的方式崩溃",这一点至关重要.这意味着"任何事情都可能发生",而且任何事情都包括看起来像程序员可能想要的那样(在台计算机上,今天用这个编译器).


作为最后一点,我没有方便访问(Valgrind,ASan,UBSan)的激进调试工具,没有足够详细的跟踪"自动"变量生命周期来捕获此错误,但是GCC 6确实产生了这个有趣的警告:

$ gcc -std=c11 -O2 -W -Wall -pedantic test.c
test.c: In function ‘main’:
test.c:9:5: warning: ‘b’ is used uninitialized in this function
    printf("%d\n", *a); // undefined behavior
    ^~~~~~~~~~~~~~~~~~

我相信这里发生的事情是,它没有我上述优化-复制的最后已知值b*a再进printf-但它的"最后已知值"的b是"这个变量是未初始化的"定点,而不是42(这然后生成相当于printf("%d\n", 0).)的代码.)


推荐阅读
  • C++字符字符串处理及字符集编码方案
    本文介绍了C++中字符字符串处理的问题,并详细解释了字符集编码方案,包括UNICODE、Windows apps采用的UTF-16编码、ASCII、SBCS和DBCS编码方案。同时说明了ANSI C标准和Windows中的字符/字符串数据类型实现。文章还提到了在编译时需要定义UNICODE宏以支持unicode编码,否则将使用windows code page编译。最后,给出了相关的头文件和数据类型定义。 ... [详细]
  • 本文介绍了深入浅出Linux设备驱动编程的重要性,以及两种加载和删除Linux内核模块的方法。通过一个内核模块的例子,展示了模块的编译和加载过程,并讨论了模块对内核大小的控制。深入理解Linux设备驱动编程对于开发者来说非常重要。 ... [详细]
  • 本文介绍了lua语言中闭包的特性及其在模式匹配、日期处理、编译和模块化等方面的应用。lua中的闭包是严格遵循词法定界的第一类值,函数可以作为变量自由传递,也可以作为参数传递给其他函数。这些特性使得lua语言具有极大的灵活性,为程序开发带来了便利。 ... [详细]
  • C语言注释工具及快捷键,删除C语言注释工具的实现思路
    本文介绍了C语言中注释的两种方式以及注释的作用,提供了删除C语言注释的工具实现思路,并分享了C语言中注释的快捷键操作方法。 ... [详细]
  • 本文介绍了C函数ispunct()的用法及示例代码。ispunct()函数用于检查传递的字符是否是标点符号,如果是标点符号则返回非零值,否则返回零。示例代码演示了如何使用ispunct()函数来判断字符是否为标点符号。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • 本文介绍了PE文件结构中的导出表的解析方法,包括获取区段头表、遍历查找所在的区段等步骤。通过该方法可以准确地解析PE文件中的导出表信息。 ... [详细]
  • C++中的三角函数计算及其应用
    本文介绍了C++中的三角函数的计算方法和应用,包括计算余弦、正弦、正切值以及反三角函数求对应的弧度制角度的示例代码。代码中使用了C++的数学库和命名空间,通过赋值和输出语句实现了三角函数的计算和结果显示。通过学习本文,读者可以了解到C++中三角函数的基本用法和应用场景。 ... [详细]
  • 本文介绍了绕过WAF的XSS检测机制的方法,包括确定payload结构、测试和混淆。同时提出了一种构建XSS payload的方法,该payload与安全机制使用的正则表达式不匹配。通过清理用户输入、转义输出、使用文档对象模型(DOM)接收器和源、实施适当的跨域资源共享(CORS)策略和其他安全策略,可以有效阻止XSS漏洞。但是,WAF或自定义过滤器仍然被广泛使用来增加安全性。本文的方法可以绕过这种安全机制,构建与正则表达式不匹配的XSS payload。 ... [详细]
  • Metasploit攻击渗透实践
    本文介绍了Metasploit攻击渗透实践的内容和要求,包括主动攻击、针对浏览器和客户端的攻击,以及成功应用辅助模块的实践过程。其中涉及使用Hydra在不知道密码的情况下攻击metsploit2靶机获取密码,以及攻击浏览器中的tomcat服务的具体步骤。同时还讲解了爆破密码的方法和设置攻击目标主机的相关参数。 ... [详细]
  • Android Studio Bumblebee | 2021.1.1(大黄蜂版本使用介绍)
    本文介绍了Android Studio Bumblebee | 2021.1.1(大黄蜂版本)的使用方法和相关知识,包括Gradle的介绍、设备管理器的配置、无线调试、新版本问题等内容。同时还提供了更新版本的下载地址和启动页面截图。 ... [详细]
  • 如何用UE4制作2D游戏文档——计算篇
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了如何用UE4制作2D游戏文档——计算篇相关的知识,希望对你有一定的参考价值。 ... [详细]
  • 成功安装Sabayon Linux在thinkpad X60上的经验分享
    本文分享了作者在国庆期间在thinkpad X60上成功安装Sabayon Linux的经验。通过修改CHOST和执行emerge命令,作者顺利完成了安装过程。Sabayon Linux是一个基于Gentoo Linux的发行版,可以将电脑快速转变为一个功能强大的系统。除了作为一个live DVD使用外,Sabayon Linux还可以被安装在硬盘上,方便用户使用。 ... [详细]
  • 本文介绍了在mac环境下使用nginx配置nodejs代理服务器的步骤,包括安装nginx、创建目录和文件、配置代理的域名和日志记录等。 ... [详细]
  • 本文介绍了一个题目的解法,通过二分答案来解决问题,但困难在于如何进行检查。文章提供了一种逃逸方式,通过移动最慢的宿管来锁门时跑到更居中的位置,从而使所有合格的寝室都居中。文章还提到可以分开判断两边的情况,并使用前缀和的方式来求出在任意时刻能够到达宿管即将锁门的寝室的人数。最后,文章提到可以改成O(n)的直接枚举来解决问题。 ... [详细]
author-avatar
赵春柱_626
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有