C语言:字符串字面量及其保存位置

相关阅读

C语言icon-default.png?t=N7T8https://blog.csdn.net/weixin_45791458/category_12423166.html?spm=1001.2014.3001.5482


        虽然C语言中不存在字符串类型,但依然可以通过数组或指针的方式保存字符串,但字符串字面量却没有想象的这么简单,本文就将对此进行讨论。

        字符串字面量保存在静态存储区,所谓静态存储区即表示其值在程序执行前就已存在于可执行文件中。如果更详细地说,字符串字面量可能保存在.rodata section中,所以当尝试使用指针改变一个字符串字面量的值时会出现段错误。需要特别说明的是上面所说的字符串字面量的位置没有任何限制,可以是在函数内,也可以是在函数外,如下面的代码所示。

例1
#include <stdio.h>
char *ptr1 = "abcde";          //"abcde"被添加到.rodata section
int main()
{char *ptr2 = "ABCDE";      //"ABCDE"被添加到.rodata sectionprintf("This is a string"); //"This is a string"被添加到.rodata section
}

        现在我们可以使用gcc编译链接成可执行程序,如下所示。

$gcc test.c -o test

        然后使用readelf命令读取二进制可执行文件test中的信息,注意这里添加了-S选项表示只输出section头的相关信息。

$readelf -S test
There are 31 section headers, starting at offset 0x36c0:Section Headers:[Nr] Name              Type             Address           OffsetSize              EntSize          Flags  Link  Info  Align[ 0]                   NULL             0000000000000000  000000000000000000000000  0000000000000000           0     0     0[ 1] .interp           PROGBITS         0000000000000318  00000318000000000000001c  0000000000000000   A       0     0     1[ 2] .note.gnu.pr[...] NOTE             0000000000000338  000003380000000000000030  0000000000000000   A       0     0     8[ 3] .note.gnu.bu[...] NOTE             0000000000000368  000003680000000000000024  0000000000000000   A       0     0     4[ 4] .note.ABI-tag     NOTE             000000000000038c  0000038c0000000000000020  0000000000000000   A       0     0     4[ 5] .gnu.hash         GNU_HASH         00000000000003b0  000003b00000000000000024  0000000000000000   A       6     0     8[ 6] .dynsym           DYNSYM           00000000000003d8  000003d800000000000000a8  0000000000000018   A       7     1     8[ 7] .dynstr           STRTAB           0000000000000480  00000480000000000000008f  0000000000000000   A       0     0     1[ 8] .gnu.version      VERSYM           0000000000000510  00000510000000000000000e  0000000000000002   A       6     0     2[ 9] .gnu.version_r    VERNEED          0000000000000520  000005200000000000000030  0000000000000000   A       7     1     8[10] .rela.dyn         RELA             0000000000000550  0000055000000000000000d8  0000000000000018   A       6     0     8[11] .rela.plt         RELA             0000000000000628  000006280000000000000018  0000000000000018  AI       6    24     8[12] .init             PROGBITS         0000000000001000  00001000000000000000001b  0000000000000000  AX       0     0     4[13] .plt              PROGBITS         0000000000001020  000010200000000000000020  0000000000000010  AX       0     0     16[14] .plt.got          PROGBITS         0000000000001040  000010400000000000000010  0000000000000010  AX       0     0     16[15] .plt.sec          PROGBITS         0000000000001050  000010500000000000000010  0000000000000010  AX       0     0     16[16] .text             PROGBITS         0000000000001060  00001060000000000000011b  0000000000000000  AX       0     0     16[17] .fini             PROGBITS         000000000000117c  0000117c000000000000000d  0000000000000000  AX       0     0     4[18] .rodata           PROGBITS         0000000000002000  000020000000000000000021  0000000000000000   A       0     0     4[19] .eh_frame_hdr     PROGBITS         0000000000002024  000020240000000000000034  0000000000000000   A       0     0     4[20] .eh_frame         PROGBITS         0000000000002058  0000205800000000000000ac  0000000000000000   A       0     0     8[21] .init_array       INIT_ARRAY       0000000000003db8  00002db80000000000000008  0000000000000008  WA       0     0     8[22] .fini_array       FINI_ARRAY       0000000000003dc0  00002dc00000000000000008  0000000000000008  WA       0     0     8[23] .dynamic          DYNAMIC          0000000000003dc8  00002dc800000000000001f0  0000000000000010  WA       7     0     8[24] .got              PROGBITS         0000000000003fb8  00002fb80000000000000048  0000000000000008  WA       0     0     8[25] .data             PROGBITS         0000000000004000  000030000000000000000018  0000000000000000  WA       0     0     8[26] .bss              NOBITS           0000000000004018  000030180000000000000008  0000000000000000  WA       0     0     1[27] .comment          PROGBITS         0000000000000000  00003018000000000000002d  0000000000000001  MS       0     0     1[28] .symtab           SYMTAB           0000000000000000  000030480000000000000378  0000000000000018          29    18     8[29] .strtab           STRTAB           0000000000000000  000033c000000000000001e1  0000000000000000           0     0     1[30] .shstrtab         STRTAB           0000000000000000  000035a1000000000000011a  0000000000000000           0     0     1
Key to Flags:W (write), A (alloc), X (execute), M (merge), S (strings), I (info),L (link order), O (extra OS processing required), G (group), T (TLS),C (compressed), x (unknown), o (OS specific), E (exclude),D (mbind), l (large), p (processor specific)

        从上面的信息中可以找到.rodata section的offset(即该section在二进制文件中的偏移)为2000。

        下面使用hexdump命令查看二进制可执行文件文件test中的内容,其中-C选项用于以十六进制单个字节的方式输出并将对应的ASCII码显示在右边。

$hexdump -C test
......
00002000  01 00 02 00 61 62 63 64  65 00 41 42 43 44 45 00  |....abcde.ABCDE.|
00002010  54 68 69 73 20 69 73 20  61 20 73 74 72 69 6e 67  |This is a string|
00002020  00 00 00 00 01 1b 03 3b  30 00 00 00 05 00 00 00  |.......;0.......|
......

        通过以上我们知道字符串字面量一般是保存在可执行文件的.rodata section中,但是否所有字符串字面量都会保存在.rodata section中呢?答案是否定的,如下例所示。

例2
char array1[6] = "aaaaaa";          //"aaaaaa"被添加到.data section作为全局数组的初始值
const char array2[6] = "bbbbbb"     //"bbbbbb"被添加到.rodata section,因为它是只读的数据
int main()
{const char array3[6] = "cccccc" //"cccccc"被添加到.rodata section,无论是否有const修饰
}

        对例2结果的验证在此跳过,交给读者进行。下面进行结果的说明,在上面的代码中,array1是一个有字符串初始值的全局数组,字符串字面量会直接添加到.data section中,即保存全局变量的section,且并不会在.rodata section保存。但array2是一个只读的有字符串初始值的全局数组,字符串字面量还是会被添加到.rodata section。至于array3,它和例1一样,无论是否有const修饰,字符串字面量都是被放在.rodata section。但要注意的是,array3与array2和array1不同,这个数组是保存在栈上的,而数组的值"cccccc"是在程序执行前就在可执行代码中的.rodata段的。在进入main函数后,栈上会开辟空间给array3,然后将.rodata区的代码复制一份到array3栈上的空间,在退出main函数后,栈上空间被自动回收。

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

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

相关文章

system.argumentnullexception值不能为null_MySQL NULL 值如何处理?

我们已经知道 MySQL 使用 SQL SELECT 命令及 WHERE 子句来读取数据表中的数据,但是当提供的查询条件字段为 NULL 时&#xff0c;该命令可能就无法正常工作。为了处理这种情况&#xff0c;MySQL提供了三大运算符:IS NULL: 当列的值是 NULL,此运算符返回 true。IS NOT NULL: 当列…

神经网络的优缺点

摘要&#xff1a; 深度学习目前受到了追捧。人们想将神经网络应用到各个地方&#xff0c;但是它们真的适合每个地方吗&#xff1f;在本篇文章中&#xff0c;将对于这个问题以及深度学习现在如此受追捧的原因进行讨论。通过这篇文章&#xff0c;您将了解神经网络的主要缺点&…

(vue基础试炼_05)简单组件之间的传值

接上一篇&#xff1a;&#xff08;vue基础试炼_04&#xff09;使用组件改造TodoListhttps://gblfy.blog.csdn.net/article/details/103844256 文章目录一、上篇回顾① 代码欣赏② 代码分析二、子组件如何向父组件传值呢&#xff1f;① 需求案例文档② 需求案例效果图③ 思考与思…

漫画:程序员战力图鉴

戳蓝字“CSDN云计算”关注我们哦&#xff01;看漫画了解程序员战斗力 are u ready&#xff1f;编程架构能力撕比嘴炮能力划水摸鱼能力花式甩锅能力画饼忽悠能力PPT吹比能力哈哈哈哈哈哈程序员哥哥上面说的是你吗 福利扫描添加小编微信&#xff0c;备注“姓名公司职位”&#xf…

Python数据挖掘与机器学习,快速掌握聚类算法和关联分析

摘要&#xff1a;前文数据挖掘与机器学习技术入门实战与大家分享了分类算法&#xff0c;在本文中将为大家介绍聚类算法和关联分析问题。分类算法与聚类到底有何区别&#xff1f;聚类方法应在怎样的场景下使用&#xff1f;如何使用关联分析算法解决个性化推荐问题&#xff1f;本…

评分9.7!这本Python书彻底玩大了?程序员:真香!

「超级星推官/每周分享」是一个围绕程序员生活、学习相关的推荐栏目。CSDN出品&#xff0c;每周发布&#xff0c;暂定5期。关键词&#xff1a;靠谱&#xff01;优质&#xff01;本期内容&#xff0c;我们将抽1人送出由我司程序员奉为“超级神作”的《疯狂Python讲义》1本&#…

vs 如何将源文件转换成可执行文件_如何将手机便签转换成word文本文档

如何将手机便签转换成word文本文档&#xff1f;实际上&#xff0c;很多手机自带的便签&#xff0c;虽然可以将单条便签内容&#xff0c;以文字或者图片的形式分享到xxx云文档&#xff0c;但却很少有支持直接转换成Word文档的。一般情况下&#xff0c;他们需要将手机便签里的内容…

端上智能——深度学习模型压缩与加速

摘要&#xff1a;随着深度学习网络规模的增大&#xff0c;计算复杂度随之增高&#xff0c;严重限制了其在手机等智能设备上的应用。如何使用深度学习来对模型进行压缩和加速&#xff0c;并且保持几乎一样的精度&#xff1f;本文将为大家详细介绍两种模型压缩算法&#xff0c;并…

jvm调优 java_opt_Java-100天知识进阶-JVM调优工具-JDK自带工具-知识铺《八》

原标题&#xff1a;Java-100天知识进阶-JVM调优工具-JDK自带工具-知识铺《八》JVM 监控分析工具一、JDK 自带工具1. jconsoleJDK/bin 目录下&#xff0c; jconsole.exe 双击打开&#xff0c; 自动搜索本机运行的虚拟机进程。选择其中一个进程即可&#xff0c;开始进行JVM监控1.…

这帮吃货程序猿,给阿里食堂来了一波骚操作

我叫宋爽&#xff0c;在别人眼里&#xff0c;我是一个程序猿。 别的程序猿&#xff0c;喜欢摁键盘&#xff0c;我嘛&#xff0c;就喜欢吃。 有一次&#xff0c;去医院体检&#xff0c;拿到CT片的我&#xff0c;看着自己的脊椎骨&#xff0c;脑子中一直在想&#xff1a;啊&#…

linux 查看shell脚本执行了多长时间

步骤说明命令①写一个1.sh脚本vim1.sh&#xff0c;把脚本内容复制进去②赋予1.sh可执行权限chmod u_x 1.sh③把要执行的命令复制到第4行&#xff0c;保存退出esc :wq④执行脚本./1sh⑤查看执行脚本&#xff0c;共消耗多长时间基础脚本模板 #!/bin/bash starttimedate %Y-%m-%d…

任正非表示华为可以向美国企业转让5G技术;谷歌同意支付5亿欧元买断与法国纠纷;京东公布iPhone11系列预售战报……...

关注并标星星CSDN云计算极客头条&#xff1a;速递、最新、绝对有料。这里有企业新动、这里有业界要闻&#xff0c;打起十二分精神&#xff0c;紧跟fashion你可以的&#xff01;每周三次&#xff0c;打卡即read更快、更全了解泛云圈精彩newsgo go go 联想举办YOGA 2019秋季新品品…

Tensorflow快餐教程(1) - 30行代码搞定手写识别

摘要&#xff1a; Tensorflow入门教程1去年买了几本讲tensorflow的书&#xff0c;结果今年看的时候发现有些样例代码所用的API已经过时了。看来自己维护一个保持更新的Tensorflow的教程还是有意义的。这是写这一系列的初心。快餐教程系列希望能够尽可能降低门槛&#xff0c;少讲…

科普帖:什么是服务网格?

戳蓝字“CSDN云计算”关注我们哦&#xff01;译| 风车云马文| Josh Fruhlinger来源| InfoWorld网站在数字化转型的背景下&#xff0c;IT行业正在将大型的应用程序集成到小的、离散的微服务容器中&#xff0c;这些容器包含所有的服务代码和依赖项&#xff0c;而这些依赖项彼此独…

年轻人,你为什么来阿里做技术?

摘要&#xff1a; 一个阿里IT男&#xff0c;成了母婴专家? 最近&#xff0c;阿里算法工程师拉普当上了爸爸。 两个月前他就开始为这件事忙活。自从老婆有喜之后&#xff0c;七大姑八大姨都来给他提供参考&#xff0c;生过孩子的朋友、同事都对他说什么该买&#xff0c;主管也时…

查一个字段中字符集超过30的列_详细解读MySQL的30条军规

一、基础规范&#xff08;1&#xff09;必须使用InnoDB存储引擎 解读&#xff1a;支持事务、行级锁、并发性能更好、CPU及内存缓存页优化使得资源利用率更高&#xff08;2&#xff09;必须使用UTF8字符集 解读&#xff1a;万国码&#xff0c;无需转码&#xff0c;无乱码风险&am…

ThinkPHP 5使用OSS

摘要&#xff1a; ThinkPHP 5使用OSS示例。准备工作登录阿里云官网&#xff0c;将鼠标移至产品找到并单击对象存储OSS打开OSS产品详情页面。在OSS产品详情页中的单击立即开通。开通服务后&#xff0c;在OSS产品详情页面单击管理控制台直接进入OSS管理控制台界面。您也可以单击位…

Linux环境_源码安装LibreOffice实现文件在线预览doc,doxc,xls,xlsx,ppt,pptx 文件

因业务需求需要&#xff0c;利用LibOffice可以打开的文档的转换。 服务器版本 环境系统版本LinuxRed Hat Enterprise Linux Server release 7.6 (Maipo)LinuxCentOS Linux release 7.6 需要准备软件包 软件及版本下载链接LibreOffice_6.3.2_Linux_x86-64_rpm.tar.gzhttps://w…

关于云原生需要了解什么

戳蓝字“CSDN云计算”关注我们哦&#xff01;尽管随处可闻云原生&#xff0c;却鲜少有人告诉你到底什么是云原生&#xff0c;若是找资料来看&#xff0c;读完大多会感觉云缭雾绕&#xff0c;一知半解&#xff0c;总之虚得很&#xff0c;甚至会让你一度怀疑自己的智商&#xff0…

仅使用NumPy完成卷积神经网络CNN的搭建(附Python代码)

摘要&#xff1a; 现有的Caffe、TensorFlow等工具箱已经很好地实现CNN模型&#xff0c;但这些工具箱需要的硬件资源比较多&#xff0c;不利于初学者实践和理解。因此&#xff0c;本文教大家如何仅使用NumPy来构建卷积神经网络&#xff08;Convolutional Neural Network , CNN&a…