作者x246libra (宸火)
看板C_and_CPP
标题[问题] 字串输出疑问
时间Fri Nov 3 15:59:05 2017
开发平台(Platform): (Ex: Win10, Linux, ...)
xp
编译器(Ex: GCC, clang, VC++...)+目标环境(跟开发平台不同的话需列出)
code block
问题(Question):
程式码交换位置後 执行出现错误
喂入的资料(Input):
1234
程式码(Code):(请善用置底文网页, 记得排版)
#include <stdio.h>
#include <stdlib.h>
main()
{
char str1[]="";
char *str2="您输入的字串如上";
printf("请输入字串:");
fgets(str1,80,stdin);
printf("%s",str1);
printf("%s",str2);
}
以上程式码 执行时 会出现错误
------------------------------------------
#include <stdio.h>
#include <stdlib.h>
main()
{
char str1[]="";
printf("请输入字串:");
fgets(str1,80,stdin);
printf("%s",str1);
char *str2="您输入的字串如上";
printf("%s",str2);
}
将 char *str2="您输入的字串如上";
这段移动到下方
就可以正常执行
不明白为什麽
--
※ 发信站: 批踢踢实业坊(ptt.cc), 来自: 61.220.35.157
※ 文章网址: https://webptt.com/cn.aspx?n=bbs/C_and_CPP/M.1509695947.A.7F8.html
1F:→ MOONRAKER: 没有为什麽,刚好而已。两段都是错的。 11/03 16:09
2F:→ x246libra: 我是按照书本练习打的 执行出现错误的是书本范例 11/03 16:11
3F:→ x246libra: 我自己移到到下方才可以执行 但是不明白为什麽 11/03 16:11
4F:→ x246libra: 为什麽说两段都是错误的? 11/03 16:12
5F:→ stupid0319: 上面的那个如果输入超过4BYTE就溢位修改到str2指标了 11/03 16:18
6F:→ stupid0319: str2指标错误之後,printf("%s",str2);就挂掉了 11/03 16:19
7F:→ stupid0319: 下面那个是str2指标在printf("%s",str1);後才补上 11/03 16:20
8F:→ stupid0319: 运气好没挂点,但是两段code都是零分XD 11/03 16:21
9F:推 ssdoz2sk: 烧书了(O 11/03 16:25
10F:→ x246libra: 但是我只输入1234 没超过4BTYE 还是执行错误 11/03 16:26
11F:→ ssdoz2sk: \0 11/03 16:27
12F:→ stupid0319: 你确定是4BYTE? 还是5BYTE,还是6BYTE? 想清楚唷 11/03 16:28
13F:→ stupid0319: 来4,5,6 BYTE选一个 11/03 16:30
14F:→ x246libra: 加上 \0 应该是 5BYTE 吧 11/03 16:33
维持 第一个程式的顺序
改为输入 A
执行不会有错误
但是
printf("%s",str2);的功能没有显示出来
输出如下=>
请输入字串:1
1
---------------------------------------
改为输入 AB
执行还是有错误 当机
这次应该没超过4 Byte了吧?
为什麽还是有错误
※ 编辑: x246libra (61.220.35.157), 11/03/2017 16:45:15
15F:→ stupid0319: 你strlen看看是不含\0几BYTE, 11/03 16:43
16F:→ stupid0319: 别在玩记忆体溢位了,骇客级别的程式设计太难 11/03 16:51
17F:→ x246libra: 用了strlen 输入1234 是得到5 11/03 16:53
18F:→ stupid0319: 5是包括\0还是不包括\0呢? 11/03 16:56
19F:→ stupid0319: 或者是包括了什麽特殊字元? 11/03 16:57
main()
{
char str1[]="";
int n;
fgets(str1,80,stdin);
printf("%d",strlen(str1));
}
单纯使用 strlen 输入1234
输出5
※ 编辑: x246libra (61.220.35.157), 11/03/2017 17:02:37
20F:推 loveflames: strlen是算str1那个字串的长度,要知道阵列不能作为 11/03 17:06
21F:→ loveflames: 函数参数,会decay成指标,然後没当的原因是你覆盖的 11/03 17:07
22F:→ loveflames: 只是int 11/03 17:08
stupid0319: 上面的那个如果输入超过4BYTE就溢位修改到str2指标了
这段话 老实说 我不太懂 不知道是否可以再说明一下
为什麽不可以超过4byte?
程式码中 fgets(str1,80,stdin); 不是说可以可以使用80个字元吗?
如果真的4BYTE就溢位 又什麽会修改到str2指标?
他们的记忆体位置有连续吗?
※ 编辑: x246libra (61.220.35.157), 11/03/2017 17:10:36
23F:→ loveflames: 看起来是32bit环境,那指标是4byte,超过对齐的范围 11/03 17:25
24F:→ loveflames: 就会覆盖到後面的 11/03 17:26
25F:→ loveflames: 这些都是undefined行为,研究这麽没营养的东西干嘛 11/03 17:27
26F:→ stupid0319: 太难了,str1因记忆体对齐的关系会配到4BYTE 11/03 17:28
请问一下 是因为这句程式 char str1[]="";
所以str1才被配到4Byte吗?
27F:→ stupid0319: 如果不是想当骇客的话,不要这样乱写程式 11/03 17:29
28F:→ loveflames: 我对这个小有涉猎,虽然是很久以前的事 11/03 17:30
只是想了解後面的原理 只是个初学者
没特别想这样写程式
※ 编辑: x246libra (61.220.35.157), 11/03/2017 17:32:38
29F:→ stupid0319: 请写char str1[80] = ""; 11/03 17:30
30F:→ stupid0319: 而fgets会吃到\n,都要算进去,你都没处理,都没教? 11/03 17:32
31F:推 stupid0319: char str1[]="";同等於 char str1[1]; strl[0] = '\0' 11/03 17:36
32F:→ loveflames: str2在32bit环境是4byte,str1不到4byte,会对齐成4by 11/03 17:36
33F:→ loveflames: te 11/03 17:36
34F:→ loveflames: 研究这个的原理也是浪费时间 11/03 17:38
谢谢以上的回答 那就不再研究下去了
至少让我知到了 不到4Byte 会自动对齐到 4Byte
感谢各位的回答
原本也只是好奇 为什麽书本说可以执行 但是我却不能
没特别想说要研究记忆体溢位
※ 编辑: x246libra (61.220.35.157), 11/03/2017 17:41:55
35F:→ loveflames: 如果不是有心细究程式在记忆体的架构,就避开这种事 11/03 17:39
36F:→ stupid0319: 有些游戏机的破解就是利用游戏存档名造成溢位 11/03 17:40
37F:→ stupid0319: 用精心设计的档名字串把程式码导向别地方 11/03 17:41
38F:→ x246libra: 我是想说 似乎人家都说C语言重视记忆体规划 11/03 17:42
39F:→ x246libra: 所以才想说多了解一点 这程式记忆体溢位是如何产生 11/03 17:43
40F:→ loveflames: 这不是C的层次,而是OS了,如果对PE或ELF有兴趣再看 11/03 17:45
41F:→ x246libra: 谢谢您的热心回答 11/03 17:48