【计算机系统基础3】数据的存储与运算

news2025/1/11 18:45:28

3.程序调试与实践:数据存储与运算

3.1真值与机器数

真值: 数据在现实世界中的表示

机器数: 数据在计算机内部的二进制编码表示

温度:零下3.5度

习惯写法:-3.5 (数据的真值/数据的实际值)

3.1.1整数的编码

带符号整数:charshortintlong

无符号整数:unsigned charunsigned shortunsigned int

  • 示例代码1

    #include<stdio.h>
    
    int main()
    { 
        int ai = 100,bi = 2147483648,ci = -100; // 2^31== 2147483648
        unsigned au = 100,bu = 2147483648,cu = -100;
        printf("ai=%d, bi=%d, ci=%d\n",ai,bi,ci);
        printf("au=%u, bu=%u, cu=%u\n",au,bu,cu);
        return 0;
    }
    
  • 示例代码1编译运行

    ./manu 
    ai=100, bi=-2147483648, ci=-100
    au=100, bu=2147483648, cu=4294967196
    

    问题:带符号整数 bi 的输出结果值为何是负数?

    无符号整数 cu,赋值-个负的数据后,cu 输出的结果为什么会是这个值?

    cu 在计算机中实际存储的内容是什么?

    image-20220613165810912

  • 反汇编查看示例1代码

    image-20220614142339789

    相同颜色为对应值

  • 问题回答

    bi 编码成为 0xffffff9c,符号位为1,为负数,所以输出为负数。

    cu 编码成为 0xffffff9c,对于无符号整数来说,换算成十进制就是 4294967196

    带符号整数: 补码image-20220614162442267

    无符号整数: 二进制编码image-20220614162450026

3.2数据的存储

  • 示例代码

    #include "stdio.h"
    void main()
    {
        char a = 100;
        short b = 100;
        int c = 100;
        int d = 0x12345678;
        printf("a=%0xH,b=%0xH,c=%0xH,d=%0xH\n", a, b, c, d);
    }
    
  • 调试

    (gdb) i r rsp rbp
    rsp            0x7ffffffedde0      0x7ffffffedde0
    rbp            0x7ffffffeddf0      0x7ffffffeddf0
    ...//执行完 int d = 0x12345678;
    (gdb) x/16xb $rsp
    0x7ffffffedde0: 0xe0    0xde    0xfe    0xff    0xff    0x64    0x64    0x00
    0x7ffffffedde8: 0x64    0x00    0x00    0x00    0x78    0x56    0x34    0x12
    

    变量在栈帧中的存储,因为系统是小端模式。

    大端方式:最高有效字节存放在低地址单元中,
    最低有效字节存放在高地址单元中。

    小端方式:最高有效字节存放在高地址单元中,
    最低有效字节存放在低地址单元中。

    image-20220614195202403

3.3数组的对齐

  • 示例代码

    #include "stdio.h"
    
    void main()
    {
        struct record
        {
            char a;
            int b;
            short c;
            char d;
        } R[2];
        R[0].a = 1;
        R[0].b = 2;
        R[0].c = 3;
        R[0].d = 4;
        R[1].a = 5;
        R[1].b = 6;
        R[1].c = 7;
        R[1].d = 8;
        printf("数据存储时的边界对齐");
    }
    
  • 调试

    ...//运行完R[1].d = 8;
    (gdb) x/32xb $rsp
    0x7ffffffeddd0: 0x01    0x00    0x00    0x00    0x02    0x00    0x00    0x00
    0x7ffffffeddd8: 0x03    0x00    0x04    0x08    0x05    0x00    0x00    0x00
    0x7ffffffedde0: 0x06    0x00    0x00    0x00    0x07    0x00    0x08    0x00
    0x7ffffffedde8: 0x00    0x8c    0x05    0x6b    0xb3    0xa8    0xe2    0x8c
    

    image-20220614204137305

image-20220614204232393

image-20220614204437791

不考虑对齐方式下:数组R占用 (1+4+2+1)x2=16 字节

对齐方式下:数组R占用 (1+3+4+2+1+1)x2=24 字节

相比于不对齐,每个数组多占用 4 个字节

  • 示例代码修改后

    #include "stdio.h"
    
    void main()
    {
        struct record
        {
            char a;
            char d;
            short c;
            int b;
        } R[2];
        R[0].a = 1;
        R[0].b = 2;
        R[0].c = 3;
        R[0].d = 4;
        R[1].a = 5;
        R[1].b = 6;
        R[1].c = 7;
        R[1].d = 8;
        printf("数据存储时的边界对齐");
    }
    
  • 调试

    (gdb) x/20xb  $rsp
    0x7ffffffeddd0: 0x01    0x04    0x03    0x00    0x02    0x00    0x00    0x00
    0x7ffffffeddd8: 0x05    0x08    0x07    0x00    0x06    0x00    0x00    0x00
    0x7ffffffedde0: 0xe0    0xde    0xfe    0xff
    

    image-20220614204739142

3.4数据类型的转换

3.4.1整数之间的数据类型转换

机器数之间的转换

赋值语句:b = a;

  • 情况一:相同宽度的两个整型数据之间的赋值

    • image-20220615131534395
    • ab 的机器数相同,真值不一定相同,取决于 ab 的数据类型
  • 情况二:将一个短的数据类型赋值给一个长的数据类型

    • image-20220615132937320

    • an01 序列复制在 b 的低 n 位,b的高 m-n 位由 a 的数据类型决定

      • n位无符号整数

      • image-20220615133053711

        将b的高 m-n 位置为0

      • n位带符号整数

      • image-20220615133311016

        将b的高 m-n 位置为 a 的符号位

  • 情况三:将一个长的数据类型赋值给一个短的数据类型

    • image-20220615133504167
    • a 的低 m 位的 01 序列赋值给 b ,丢弃 a 的高位部分
  • 示例代码

    #include "stdio.h"
    void main()
    {
        short si = -100;
        unsigned short usi = si;
        int i = usi;
        unsigned ui = usi;
        int i1 = si;
        unsigned ui1 = si;
        int i2 = 0x12348765;
        short si2 = i2;
        unsigned short usi2 = i2;
        int i3 = si2;
        int i4 = 4294967296;
        printf("si=%d,usi=%u,i=%d,ui=%u,i1=%d,ui1=%u\n", si, usi, i, ui, i1, ui1);
        printf("i2=%d,si2=%d,usi2=%u,i3=%d,i4=%d\n", i2, si2, usi2, i3, i4);
    }
    
  • 调试

    • 相同宽度的两个整型数据之间的赋值示例

      image-20220615140140028

    • 零扩展代码示例

      image-20220615140252629

    • 符号扩展示例

      image-20220615140616457

    • 截断

      image-20220615140756018

      image-20220615140917587

3.4.2整数与浮点数之间的转换

  • 示例代码

    #include<stdio.h>
    int main()
    {
        int i1=0x7fffffff,i2,itemp;
        float f1=0x987654321,f2,ftemp;
        ftemp = i1;
        i2 = ftemp;
        itemp = f1;
        f2 = itemp;
        printf("i1=%d,i2=%d,f1=%f,f2=%f\n",i1,i2,f1,f2);
        return 0;
    }
    
  • 整数到浮点数

    要进行数据编码格式上的转换,而不是机器数上的直接复制。

    image-20220622211902528

    image-20220623100547231

    没看太明白😕

  • 浮点数到整数

    image-20220623101820282

3.4.3自动类型转换

  • 示例代码

    #include<stdio.h>
    
    int f1(unsigned int n)
    {
        int sum = 1,power = 1;
        int i;
        for(i = 0;i <= n - 1;i++)
        {
            power *= 2;
            sum += power;
        }
        return sum;
    }
    
    int main()
    {
        int sum;
        sum = f1(0);
        printf("sum=%d\n",sum);
        return 0;
    }
    
  • 执行后结果:死循环

    image-20220623111455271

    关键在于这儿的比较,in - 1 的比较会自动转换为无符号整型的比较,执行完 n - 1 后,n的数值(edx)如下

    image-20220623111644596

    这个数在无符号整型的比较下恒成立,所以为死循环

  • 修改内容

    #include<stdio.h>
    
    int f1(int n)//改为整型
    {
        int sum = 1,power = 1;
        int i;
        for(i = 0;i <= n - 1;i++) 
        {
            power *= 2;
            sum += power;
        }
        return sum;
    }
    
    int main()
    {
        int sum;
        sum = f1(0);
        printf("sum=%d\n",sum);
        return 0;
    }
    

3.5浮点数的表示和运算–IEEE 754

3.5.1 IEEE 754 浮点数基本格式

  • float 类型,32 位的单精度浮点格式

    image-20220627144642691

  • double 类型,64 位的双精度浮点格式

    image-20220627150543133

3.5.2 IEEE 754 数据按置的分类(float 为例)

  • 分类五类,下图没有体现的是无定义数

image-20220627150835020

  • 各类值的编码

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Utc5JczA-1684569046231)(…/软件杯-高性能分析型连接查询/image-20220627151052118.png)]

  • 真值与机器数对应关系举例

    • 规格化数

      image-20220627155517620

    • 非规格化数

      image-20220627155618373

  • 示例程序

    #include "stdio.h"
    void main()
    {
        float finf1 = 4e38,finf2 = 5e38,finf3 = 6e38;         //正无穷大
        float fninf1 = -4e38, fninf2 = -5e38, fninf3 = -6e38; //负无穷大
        float fzero = 0, fnzero = -fzero;// 0,-0
        float fnormal1 = 5.0, fnormal2 = 0.1, fnnormal1 = -5, fnnormal2 = -0.1; //规格化数
        float ffrac = 1e-40, fnfrac = -1e-40;//非规格化数
        float fnan1 = finf1 + fninf1, fnan2 = -fnan1;//无定义数
        float finf4 = fnormal1 / fzero; //除以0,无穷大
        printf("%f %f %f\n", finf1, finf2, finf3);
        printf("%f %f %f\n", fninf1, fninf2, fninf3);
        printf("%f %f\n", fzero, fnzero);
        printf("%f %.20f \n%f %.20f\n", fnormal1, fnormal2, fnnormal1, fnnormal2);
        printf("%.50f\n%.50f\n", ffrac, fnfrac);
        printf("%f %f\n", fnan1, fnan2);
        printf("%f \n", finf4);
    }
    
  • 输出

    image-20220627164938665

  • 打印结果分析

    • 第一行

      image-20220627165028737

    • 第二行

      image-20220627165202867

    • 第三行

      image-20220627165230961

    • 第四行

      image-20220627165339065

    • 第五行

      image-20220627165430826

    • 第六行

      image-20220627165502995

    • 第七行

      image-20220627165624001

3.6浮点数的表示和运算–尾数的舍入处理

  • 以32位单精度浮点数格式为例(就近舍入法)

    image-20220627170958401

    image-20220627171408845

  • 示例代码

    #include <stdio.h>
    
    void main()
    {
        float a1 = 0x8000000, a2 = 0x8000001, a3 = 0x8000014, a4 = 0x8000017;
        float b1 = 0x8000019, b2 = 0x800000c, b3 = 0x800000d;
        float c1 = 0x8000008, c2 = 0x8000018;
        float g = 0.1;
        printf("a1=%xH,a2=%xH,a3=%xH,a4=%xH\n",(int)a1,(int)a2,(int)a3,(int)a4);
        printf("b1=%xH,b2=%xH,b3=%xH\n",(int)b1,(int)b2,(int)b3);
        printf("c1=%xH,c2=%xH\n",(int)c1,(int)c2);
        printf("g=%.20f",g);
    };
    
  • 编译运行

     ./floatround 
    a1=8000000H,a2=8000000H,a3=8000010H,a4=8000010H
    b1=8000020H,b2=8000010H,b3=8000010H
    c1=8000000H,c2=8000020H
    g=0.10000000149011611938
    

    a1. a2、a3、a4:输出值 ≤ 初始值, 舍操作
    b1、b2、b3:输出值 > 初始值, 入操作
    c1:输出值 < 初始值, 舍操作
    c2:输出值 > 初始值, 入操作
    g:输出值> 0.1, 0.1用机器数不可以精确表示,入操作

  • 分析

    • image-20220627173444480

    • image-20220627173506973

    • image-20220627173533371

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

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

相关文章

15:13进去面试,5分钟就完事了,问的实在是太......

干了两年外包&#xff0c;本来想出来正儿八经找个互联网公司上班&#xff0c;没想到算法死在另一家厂子。 自从加入这家外包公司&#xff0c;每天都在加班&#xff0c;钱倒是给的不少&#xff0c;所以也就忍了。没想到11月一纸通知&#xff0c;所有人不许加班&#xff0c;薪资…

2023年值得关注的低代码平台推荐

低代码平台在数字化转型的浪潮中受到越来越多企业的青睐&#xff0c;因为它们提供了一种更容易、更快的方式来开发网络和移动应用程序。低代码平台只需要最少的编码知识&#xff0c;使公司能够在很短的时间内开发出定制的应用程序&#xff0c;而这只是使用传统的搭建手段所需时…

参数传递之传名,传地址,得结果,传值

编译原理速成&#xff0c;参数传递之传名&#xff0c;传地址&#xff0c;得结果&#xff0c;传值(四)_哔哩哔哩_bilibili 学习自上面的文章。 题目1&#xff1a; &#xff08;1&#xff09;传名。 (2)传地址。&#xff08;注意观察&#xff0c;AB变成了临时变量T1&#xff0c;…

jvm之G1 GC

写在前面 jdk9以及之后的版本已经将默认的垃圾收集器parallel更换为G1.本文就一起来看下。 1&#xff1a;G1介绍 parallel GC的设计目标是高吞吐量&#xff0c;CMS GC的设计目标是低延迟&#xff0c;而G1的设计目标不是这二者中的任何一个&#xff0c;其设计目标是让GC的STW…

我的世界Fabric mod开发-快速漏斗

前往我的主页以阅读完整内容&#xff0c;并获取源码 DearXuan的主页 MOD介绍 使用漏斗链进行分类或传递物品时,常常会发现漏斗速度太慢,难以收集全部掉落物.或者漏斗太多,影响性能.而现有的漏斗加速mod则是引入新的快速漏斗,存在各种兼容问题.开服时发现paper服务器可以修改原…

华为OD机试真题 Java 实现【区间连接器】【2023Q1 200分】

一、题目描述 有一组区间 [a0, b0], [a1, b1], … (a, b 表示起点, 终点)&#xff0c;区间有可能重叠、相邻&#xff0c;重叠或相邻则可以合并为更大的区间&#xff1b; 给定一组连接器[x1, x2, x3, …]&#xff08;x 表示连接器的最大可连接长度&#xff0c;即 x>gap&…

支付宝沙箱支付(java电脑版)

目录 下载支付demo配置环境AlipayConfig 下载支付demo 网址&#xff1a;https://open.alipay.com/ 下载并打开项目发现无法运行&#xff1a; 手动转化项目&#xff1a; 等待下载整理一下maven pom 通过tomat部署运行测试。 导入阿里支付的pom依赖 <dependency> &l…

都2023了,你竟然还不知道网络安全该怎么学

前言 网络安全是指网络系统的硬件、软件及其系统中的数据受到保护&#xff0c;不因偶然或恶意原因而遭受破坏、更改、泄露&#xff0c;系统连续可靠正常地运行&#xff0c;网络服务不中断。网络安全因何而重要&#xff1f; 截至2023年4月,我国网民规模为_11.51亿_&#xff0c…

模板初阶(泛型编程)

模板初阶 &#x1f506;泛型编程&#x1f506;函数模板函数模板概念函数模板格式函数模板的原理函数模板的实例化模板参数的匹配原则 &#x1f506;类模板类模板的定义格式类模板的实例化类模板与模板类的区别 &#x1f506;结语 &#x1f506;泛型编程 泛型编程&#xff1a;编…

【总结】Numpy2

Numpy 1. 数组和数的运算 array1 np.arange(1,10) array1 # array([1, 2, 3, 4, 5, 6, 7, 8, 9]) array1 10 # array([11, 12, 13, 14, 15, 16, 17, 18, 19]) array1 - 10 # array([-9, -8, -7, -6, -5, -4, -3, -2, -1]) array1 * 10 # array([10, 20, 30, 40, 50, 60, 70…

Flutter:如何在Android中实现串口通信调试

本文介绍如何通过flutter_libserialport插件在Flutter中实现串口通信调试。 1、引入依赖 在flutter工程的pubspec.yaml文件中引入flutter_libserialport依赖&#xff1a; dependencies:flutter_libserialport: ^0.3.0 2、导入import依赖包 在dart代码中import导入flutter_li…

快手国际化 后端开发面经二面

目录 1.Redis用的什么数据类型2.Hash底层结构3.JVM垃圾判别阶段算法4.MySQL索引模型5.为什么用B树6.联合索引在B树如何构造的7.覆盖索引知道吗 1.Redis用的什么数据类型 1.String(字符类型) 2.Hash(散列类型) 3.List(列表类型) 4.Set(集合类型) 5.SortedSet(有序集合类型&…

设计模式 单例模式(创建型)

一、前言 学习设计模式我们关注的是什么&#xff0c;如何实现么&#xff1f;是也不是。我认为比了解如何实现设计模式更重要的是这些设计模式的应用场景&#xff0c;什么场景下我们该用这种设计模式&#xff1b;以及这些设计模式所包含的思想&#xff0c;最终帮助我们把代码写…

继承 + 多态 + final + 权限修饰符

目录 继承 多态 final 权限修饰符 继承 继承定义&#xff1a; 可以让类跟类之间产生子父的关系继承的好处 可以把多个子类中重复的代码抽取到父类中&#xff0c;子类可以直接使用&#xff0c;减少代码几余&#xff0c;提高代码的复用性子类继承内容 非私有private构造方法…

#机器学习--深度学习中的正则化

#机器学习--深度学习中的正则化 引言1、参数范数惩罚2、 L 2 L^{2} L2 正则化3、 L 1 L^{1} L1 正则化4、显式约束和重投影5、参数绑定和参数共享6、Bagging7、Dropout 引言 本系列博客旨在为机器学习(深度学习)提供数学理论基础。因此内容更为精简&#xff0c;适合二次学习的…

uniapp实现条码扫描 可开闪光灯,原生H5调用,不需要任何sdk。

主要思路 使用QuaggaJs这个库。调用摄像头使用的 navigator.mediaDevices.getUserMedia 这个H5的api。通过 video 和 canvas 把摄像头获取到的数据展现到页面上&#xff0c;同时调用监听Quagga解析。 获取设备摄像头权限,用于后续开启摄像头。创建video元素显示摄像头画面,和ca…

AcWing算法提高课-1.3.10混合背包问题

宣传一下算法提高课整理 <— CSDN个人主页&#xff1a;更好的阅读体验 <— 本题链接&#xff08;AcWing&#xff09; 点这里 题目描述 有 N N N 种物品和一个容量是 V V V 的背包。 物品一共有三类&#xff1a; 第一类物品只能用1次&#xff08;01背包&#xff0…

opencv相机标定

当你把摄像机放在一个特定的位置&#xff0c;在它的后面放一个目标图像&#xff0c;或者是把摄像机放到某个物体上&#xff0c;摄像机周围的物体是什么形状&#xff0c;你需要知道这些信息。 当你在计算机上处理图像时&#xff0c;会使用以下三个参数&#xff1a; 1.像素坐标&a…

软件测试5年了,薪资25K,我还怕00后卷吗?

前言 沅哥在这个行业爬摸滚打5年了&#xff0c;从最开始点点点的功能测试到现在到现在成为高级测试&#xff0c;工资也翻了几倍&#xff0c;到了25k。他和我说&#xff0c;最开始他是想躺平的&#xff0c;后来也是被迫卷的&#xff0c;但好在这个结果他很满意。 之所以改变的…

一文3000字从0到1使用Selenium进行自动化测试

对于很多刚入门的测试新手来说&#xff0c;大家都将自动化测试作为自己职业发展的一个主要阶段。可是&#xff0c;在成为一名合格的自动化测试工程师之前&#xff0c;我们不仅要掌握相应的理论知识&#xff0c;还要进行大量的实践&#xff0c;积累足够的经验&#xff0c;以便快…