搜索
bottom↓
回复: 17
打印 上一主题 下一主题

请大家分析一下,这段C代码的输出结果是什么?

[复制链接]

出0入18汤圆

跳转到指定楼层
1
发表于 2024-4-18 21:26:31 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
#include <stdio.h>
float Test;

int main(void) {
    unsigned char Out  = 0xff;
    Test = (float)(((signed short int )Out) << 8);
    printf("Test:%f!\n", Test);
    return 0;
}

出200入2554汤圆

2
发表于 2024-4-18 21:43:19 来自手机 | 只看该作者
有符号左移啊,目测编译器相关

出0入984汤圆

3
发表于 2024-4-18 21:51:07 | 只看该作者
t3486784401 发表于 2024-4-18 21:43
有符号左移啊,目测编译器相关
(引用自2楼)


左移应该是会忽略符号位,右移参与

出0入131汤圆

4
发表于 2024-4-18 22:00:22 | 只看该作者

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x

出0入984汤圆

5
发表于 2024-4-18 22:03:47 | 只看该作者

不过经过运算符后会成为int
(float)((signed short int )(((signed short int )Out) << 8));

这样才是-256

出0入18汤圆

6
 楼主| 发表于 2024-4-18 22:04:39 | 只看该作者
我用clion测出来的结果居然是正数

出0入59汤圆

7
发表于 2024-4-18 22:41:28 | 只看该作者


METOR AI 结果如上,简洁明了。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x

出0入18汤圆

8
 楼主| 发表于 2024-4-18 23:08:46 来自手机 | 只看该作者
sonna 发表于 2024-4-18 22:41
用METOR AI 结果如上,简洁明了。
(引用自7楼)

你用clion试下
应该二楼是正解

出200入2554汤圆

9
发表于 2024-4-18 23:39:17 | 只看该作者
本帖最后由 t3486784401 于 2024-4-18 23:40 编辑

找了手头几个编译器、运行平台跑了下。 啥都别说了,这代码质量堪忧。

ICCAVR +m328 (ICC, 8bit AVR):


Arduino +m328 (GCC, 8bit-AVR):


VS2005 +Win32 (VS, Win32):


Arduino +Esp8266 (GCC, 32bit-XTENSA):



总结下来就是,8bit 平台会忽略 signed short int,移位时直接按无符号硬怼;
32bit 平台会把 signed short int 升格为 signed int32,然后就是更大的整数了。

一般很少用 signed 移位,就是避免这种坑;

在 AVR 的指令里右移有三种 ASR  ROR  LSR,左移只有两种 ROL  LSL.
缺少的那个 ASL(算数左移)就是一楼位置的代码功能,显然是个指令制定者都不愿意面对的问题。

实际在 C 里升格降格都理不清的话,更别说底层指令了。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x

出0入18汤圆

10
 楼主| 发表于 2024-4-19 06:40:55 来自手机 | 只看该作者
t3486784401 发表于 2024-4-18 23:39
找了手头几个编译器、运行平台跑了下。 啥都别说了,这代码质量堪忧。

ICCAVR +m328 (ICC, 8bit AVR):

(引用自9楼)

32位平台扩展成signed int也说不通啊,带符号扩展应该值不会变啊

出0入59汤圆

11
发表于 2024-4-19 08:54:17 | 只看该作者
tang_qianfeng 发表于 2024-4-18 23:08
你用clion试下
应该二楼是正解
(引用自8楼)

要这么玩是吧,看看下面的提问方式和答案


METOR AI还是可以的。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x

出0入18汤圆

12
 楼主| 发表于 2024-4-19 09:00:16 | 只看该作者
sonna 发表于 2024-4-19 08:54
要这么玩是吧,看看下面的提问方式和答案


(引用自11楼)

可以啥啊,回答的两个都有问题,第一个,都是8位的,左移8位后不是为0x00了么,怎么可能还会是0xff?
第二个,0xff00是有16位带符号数,怎么可能类型转换为float后会变成正数?

出0入16汤圆

13
发表于 2024-4-19 09:01:42 | 只看该作者
我认为是65280,左移完成为0xFF00,float强制转化就是65280

出0入16汤圆

14
发表于 2024-4-19 09:17:28 | 只看该作者
本帖最后由 初音之恋 于 2024-4-19 09:20 编辑
tang_qianfeng 发表于 2024-4-19 09:00
可以啥啊,回答的两个都有问题,第一个,都是8位的,左移8位后不是为0x00了么,怎么可能还会是0xff?
第二 ...
(引用自12楼)


左移8位后不一定还是int16,除非(float)((int16)(((int16)out)<<8)))=-256

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x

出0入18汤圆

15
 楼主| 发表于 2024-4-19 09:24:33 | 只看该作者
初音之恋 发表于 2024-4-19 09:01
我认为是65280,左移完成为0xFF00,float强制转化就是65280
(引用自13楼)

左移前不是已经强制类型转换为singend int了么

出0入18汤圆

16
 楼主| 发表于 2024-4-19 09:26:10 | 只看该作者
本帖最后由 tang_qianfeng 于 2024-4-19 09:30 编辑
初音之恋 发表于 2024-4-19 09:17
左移8位后不一定还是int16,除非(float)((int16)(((int16)out)
(引用自14楼)


那((int16)pp), 这个类型转换后的数据是啥类型的?结果是多少?
int16的数据左移8位,类型就有可能就不是int16了呵?

出0入16汤圆

17
发表于 2024-4-19 09:30:57 | 只看该作者
本帖最后由 初音之恋 于 2024-4-19 09:51 编辑
tang_qianfeng 发表于 2024-4-19 09:24
左移前不是已经强制类型转换为singend int了么
(引用自15楼)


对,做以前是int16,但是左移完成后就不一定是int16,因为值溢出了,比如C#这种 int16 pp, byte ff=0xff; pp=(int16)ff<<8就会报错 ,而且这种情况下 pp=(int16)((int16)ff<<8)可以正常执行,但如果pp=convert.toInt16((int16)ff<<8)编译不报错,程序直接崩溃,因为数据溢出

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x

出200入2554汤圆

18
发表于 2024-4-19 10:16:33 | 只看该作者
tang_qianfeng 发表于 2024-4-19 06:40
32位平台扩展成signed int也说不通啊,带符号扩展应该值不会变啊
(引用自10楼)

不是说不说得通,是 VS 调试界面明确发生了位扩展。

并且在发生位扩展的时候,根本就是按照无符号数进行操作的。
AVR 指令里唯独缺少算数左移,应该就是发生带符号溢出时,数学上没有个统一的标准,索性就不做这个指令。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x
回帖提示: 反政府言论将被立即封锁ID 在按“提交”前,请自问一下:我这样表达会给举报吗,会给自己惹麻烦吗? 另外:尽量不要使用Mark、顶等没有意义的回复。不得大量使用大字体和彩色字。【本论坛不允许直接上传手机拍摄图片,浪费大家下载带宽和论坛服务器空间,请压缩后(图片小于1兆)才上传。压缩方法可以在微信里面发给自己(不要勾选“原图),然后下载,就能得到压缩后的图片】。另外,手机版只能上传图片,要上传附件需要切换到电脑版(不需要使用电脑,手机上切换到电脑版就行,页面底部)。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|Archiver|amobbs.com 阿莫电子技术论坛 ( 粤ICP备2022115958号, 版权所有:东莞阿莫电子贸易商行 创办于2004年 (公安交互式论坛备案:44190002001997 ) )

GMT+8, 2024-5-2 12:34

© Since 2004 www.amobbs.com, 原www.ourdev.cn, 原www.ouravr.com

快速回复 返回顶部 返回列表