C_and_CPP 板


LINE

开发平台(Platform): (Ex: VC++, GCC, Linux, ...) GCC 问题(Question): printf的结果会根据型态的不同而改变 喂入的资料(Input): int a = 5000; char b = (char)a; printf("b = %x\n", b); 预期的正确结果(Expected Output): b = 88 (5000 = 0x1388) 错误结果(Wrong Output): b = ffffff88 补充说明(Supplement): 尝试了一些a的初始值和结果,有点不太明白为什麽会变成这样,整理如下: int a = 5000; --> b = ffffff88 int a = 0x1234; --> b = 34 int a = 50; --> b = 32 // 这个很正常 char a = 50; --> b = 32 // 这个很正常 char a = 0x7F; --> b = 7f char a = 0x80; --> b = ffffff80 char a = 0x81; --> b = ffffff81 主要是前两个,为什麽一个会印出ffffff,一个就不会? --



※ 发信站: 批踢踢实业坊(ptt.cc), 来自: 60.250.31.103
※ 文章网址: https://webptt.com/cn.aspx?n=bbs/C_and_CPP/M.1455519100.A.9CF.html
1F:→ longlongint: char的范围是? 02/15 15:05
2F:→ andrenvq57: 为什麽不是0x000088而是0xffff88 overflow是直接用ff 02/15 15:17
3F:→ andrenvq57: 顶替吗 02/15 15:17
4F:→ stupid0319: char* b = (char*)&a; 这样比较理想 02/15 15:20
5F:→ stupid0319: 结果不是b = ffffff88吗,怎麽下一行又变b = ffffff34 02/15 15:23
※ 编辑: vvrr (60.250.31.103), 02/15/2016 15:37:04
6F:→ vvrr: 是88没错@@ 抱歉 02/15 15:37
7F:推 stupid0319: char = 0x7F ,0x80 , 0x81 各试看看吧 02/15 15:42
8F:→ vvrr: char就,一般的char. 应该是-128~127 02/15 15:43
9F:→ vvrr: 2F,我也不知道为什麽是ff,gcc印出来的.. 02/15 15:44
※ 编辑: vvrr (60.250.31.103), 02/15/2016 15:46:25
10F:推 CoNsTaR: 因为你 printf 要求的输入不是 char 你却给他 char 吧… 02/15 15:45
11F:推 CoNsTaR: 它需要一个比 char 更大的型态 02/15 15:47
12F:→ CoNsTaR: 像你这里输出的结果是 4*8=32bit 的型态 02/15 15:47
13F:→ vvrr: 如果前面都会被补成ffff就还好,但是有些就不会 02/15 15:48
※ 编辑: vvrr (60.250.31.103), 02/15/2016 15:49:09
14F:→ CoNsTaR: int i = 5000; 02/15 15:49
15F:→ CoNsTaR: char c = (char)i;i = c;printf("%p\n", i); 02/15 15:49
16F:→ CoNsTaR: @vvrr因为你不能知道你读超过b之外的值是多少啊 02/15 15:50
17F:→ CoNsTaR: 如果b前面的变数刚好都是0那读出来当然就不会有f 反之 02/15 15:52
18F:→ CoNsTaR: 亦然 02/15 15:52
19F:→ CoNsTaR: 这里没有 overflow 只有 overbound 02/15 15:53
20F:推 stupid0319: FFFFFF不是补上去的吧......看了我都快吐血了 02/15 16:04
21F:推 stupid0319: 神人来说一下char跟int的负数怎麽表示吧 02/15 16:07
22F:→ stupid0319: char跟int的-1各为FF跟FFFFFFFF 02/15 16:11
23F:→ stupid0319: -2呢,FE跟FFFFFFFE 02/15 16:11
24F:→ stupid0319: 所以char的0x88跟int的0xffffff88是等值的 02/15 16:13
25F:→ stupid0319: 在一些程式中,输入0x80000000可能造成bug 02/15 16:15
26F:→ stupid0319: 很多游戏的洗钱BUG就是这麽来的 02/15 16:15
27F:推 LPH66: 关键字: sign extension 02/15 16:19
28F:→ besmartAE: (unsigned char)才对 02/15 16:21
29F:→ stupid0319: 一个道具10万块钱好了,买21475个,变成0x80003FE0 02/15 16:21
30F:→ stupid0319: 买了道具後系统还要付给你21亿 02/15 16:22
31F:→ vvrr: 「所以char的0x88跟int的0xffffff88是等值的」 02/15 16:36
32F:→ vvrr: 所以printf会先把後面的数字转成int吗? 02/15 16:36
33F:→ vvrr: ^^^^^^^^^^ 後面的char型态的b 02/15 16:37
34F:推 stupid0319: %x Unsigned hexadecimal integer 02/15 16:40
35F:推 apologize: unsigned char b = (char)a; 改成 02/15 18:21
36F:推 LPH66: >vvrr 16:36 是, 不过不是 printf 转的 02/15 22:26
37F:→ LPH66: 而是因为 printf 属於可变参数函式, 不到 int 等级的整数 02/15 22:26
38F:→ LPH66: 规定要转成 int 再传进去, 所以在那时就已经转了 02/15 22:27
39F:→ LPH66: 也因为规定转成 int, 所以会转成一个有号整数 02/15 22:27
40F:→ LPH66: 这才用上了我上面讲的 sign extension 02/15 22:28
41F:→ LPH66: 概念上就是如 stupid0319 讲的, 0x88 (等於十进位 -120) 02/15 22:29
42F:→ LPH66: 会变成 int 的 -120 (0xffffff88) 02/15 22:29
43F:→ LPH66: 那因为二进位观点来看就是最高位的正负号位元往前补满 02/15 22:29
44F:→ LPH66: 所以要说「ffffff 是补上去的」技术上来说也没有错就是了 02/15 22:30
45F:推 azter: 其实可以开小算盘的程式设计师模式 02/16 01:00
46F:→ azter: 第一步计算5000+128 02/16 01:02
47F:→ azter: 第二步 将第一步结果 mod 256 02/16 01:03
48F:→ azter: 第三步 将第二步的结果再减去减去128 02/16 01:04
49F:→ azter: 小算盘的输出结果是-120 02/16 01:07
50F:→ azter: 单看-120可能看不出端倪,请看小算盘下面显示一堆1 0那栏 02/16 01:10
51F:推 LPH66: 要讲型态转换的话, 这样操作: (1) 左下角选 dword, 10 进位 02/16 03:30
52F:→ LPH66: 然後输入 5000; (2) 左下角点选 byte; 这等同於转型成 char 02/16 03:31
53F:→ LPH66: 你会看到它变成了 -120 了 02/16 03:31
54F:→ LPH66: (3) 根据我上面说的, 传进 printf 前会再转成 int 02/16 03:31
55F:→ LPH66: 所以再点回 dword, 你会看到数值还是 -120 02/16 03:32
56F:→ LPH66: 但下面的二进位显示部份前面却是全部补了 1 进去 02/16 03:32
57F:→ LPH66: (4) 输出成 %x, 所以点选 16 进位, 就看到 ffffff88 出来了 02/16 03:33
58F:→ LPH66: 你把你实验的值代换掉上面的 5000, 观察下面二进位显示 02/16 03:34
59F:→ LPH66: 就会知道为什麽有些数会这样变有些数会那样变 02/16 03:34
60F:→ vvrr: 谢谢大家<(_ _)> 02/16 10:58
61F:→ vvrr: 1. int a 转成 char b的时候,不论正负只留最後1个byte 02/16 10:59
62F:→ vvrr: 2. char b传进printf前会根据b此时代表数值转成signed int 02/16 11:00
63F:→ vvrr: 3. printf实际上印出来的都是int.有些只看到1byte的只是前面 02/16 11:01
64F:→ vvrr: 都是0(而且我没有叫printf印出来) 大概是这样没错吧 02/16 11:02







like.gif 您可能会有兴趣的文章
icon.png[问题/行为] 猫晚上进房间会不会有憋尿问题
icon.pngRe: [闲聊] 选了错误的女孩成为魔法少女 XDDDDDDDDDD
icon.png[正妹] 瑞典 一张
icon.png[心得] EMS高领长版毛衣.墨小楼MC1002
icon.png[分享] 丹龙隔热纸GE55+33+22
icon.png[问题] 清洗洗衣机
icon.png[寻物] 窗台下的空间
icon.png[闲聊] 双极の女神1 木魔爵
icon.png[售车] 新竹 1997 march 1297cc 白色 四门
icon.png[讨论] 能从照片感受到摄影者心情吗
icon.png[狂贺] 贺贺贺贺 贺!岛村卯月!总选举NO.1
icon.png[难过] 羡慕白皮肤的女生
icon.png阅读文章
icon.png[黑特]
icon.png[问题] SBK S1安装於安全帽位置
icon.png[分享] 旧woo100绝版开箱!!
icon.pngRe: [无言] 关於小包卫生纸
icon.png[开箱] E5-2683V3 RX480Strix 快睿C1 简单测试
icon.png[心得] 苍の海贼龙 地狱 执行者16PT
icon.png[售车] 1999年Virage iO 1.8EXi
icon.png[心得] 挑战33 LV10 狮子座pt solo
icon.png[闲聊] 手把手教你不被桶之新手主购教学
icon.png[分享] Civic Type R 量产版官方照无预警流出
icon.png[售车] Golf 4 2.0 银色 自排
icon.png[出售] Graco提篮汽座(有底座)2000元诚可议
icon.png[问题] 请问补牙材质掉了还能再补吗?(台中半年内
icon.png[问题] 44th 单曲 生写竟然都给重复的啊啊!
icon.png[心得] 华南红卡/icash 核卡
icon.png[问题] 拔牙矫正这样正常吗
icon.png[赠送] 老莫高业 初业 102年版
icon.png[情报] 三大行动支付 本季掀战火
icon.png[宝宝] 博客来Amos水蜡笔5/1特价五折
icon.pngRe: [心得] 新鲜人一些面试分享
icon.png[心得] 苍の海贼龙 地狱 麒麟25PT
icon.pngRe: [闲聊] (君の名は。雷慎入) 君名二创漫画翻译
icon.pngRe: [闲聊] OGN中场影片:失踪人口局 (英文字幕)
icon.png[问题] 台湾大哥大4G讯号差
icon.png[出售] [全国]全新千寻侘草LED灯, 水草

请输入看板名称,例如:Soft_Job站内搜寻

TOP