【C语言进阶深度学习记录】三 浮点数(float) 在内存中的表示方法

  • 相信大多数人知道整形数在内存中的分布方式,而且也能很容易写出其二进制的形式,但是对于浮点数,估计知道的人并不是很多
  • 今天学习在C语言中浮点数在内存中的表示方法

文章目录

    • 1 浮点数在内存中的存储方式
      • 1.1 浮点数的转换步骤
      • 1.2 浮点数的转换实际例子分析
      • 1.3 编程验证测试
    • 2 int与float类型的范围的比较
      • 2.1 float数不精确的编码案例
    • 3 总结

1 浮点数在内存中的存储方式

  • 浮点数在内存中的存储方式为:符号位+指数+尾数。对于float与double类型的存储方式,如下图所示:

在这里插入图片描述

  • float与double类型的数据在计算机内部的表示法是相同的,但是由于所占存储空间的不同,其分别能表示的数值范围和精度不同。

上面的表示法,可能并不是很好理解,看到后面的例子就会恍然大悟。

1.1 浮点数的转换步骤

具体如何将浮点数转换成内存中的二进制(或者十六进制的形式),下面就来看看转换步骤。

  1. 首先将浮点数转换成二进制形式(这个自己查资料学习如何转换,不是本文的目的)
  2. 用科学计数法表示二进制浮点数
  3. 计算指数偏移后的值
  • 注意,第三点需要明白,计算指数时需要加上偏移量(这里不讲解为什么存在偏移量,后序文章会学习),对于float与double类型,加上的偏移量是不一样的。具体如下:

对于指数为6,偏移后的值为:

float:6+127 ---->> 133
double : 6+1023 ---->> 1029

对于上面的内容,目前可能还没有理清。但是下面的例子,可以将上面的所有内容学习明白

1.2 浮点数的转换实际例子分析

上面说了一堆,这里来一个例子彻底明白上述的转换规则。

  • 实数8.25在内存中的float表示为如下:
  1. 首先是确定8.25的二进制表示形式:1000.01

  2. 然后用科学技术法表示二进制浮点数:1.00001 * (23).

  3. 计算符号位:3+127 = 130 ----> 10000010

  4. 符号位为0, 10000010 , 小数为(尾数):00001

由上面4条的计算以及前面的规则,得到8.25的float表示为:

  • 0 10000010 00001000000000000000000 ----> 0x41040000

至此,已经得出了8.25的浮点数表示形式。对于double的表示形式,可以自己进行推导。

1.3 编程验证测试

对于1.2小节的内容,可以编写下面的程序来进行验证:

#include <stdio.h>int main()
{float f = 8.25;unsigned int* p = (unsigned int*)&f;printf("0x%08X\n", *p);return 0;
}

运行结果也是等于: 0x41040000 可以验证上述规则与推导的正确性。

2 int与float类型的范围的比较

我们知道int与float都是占用4字节内存,但是他们所能够表示的数的范围并不是一样的。如下图:

  • int 类型的范围:[-231 , 231]
  • float类型的范围:[-3.4 * 1038 , 3.4 * 1038]

首先抛开float为什么是这个范围不说(具体的原因本文不讲)。为什么int和float同样是占用4字节的内存,但是float所表示的数的范围明显比int的范围大很多?

其实只要占用的字节数相同,那么所能表示的数的个数,就一定相等,比如一字节的内存就最多能表示 256个数字。但是有符号和无符号表示的数范围就不一样。注意,是数范围不一样,但是他们都只能表示256个数字。只是这256个数字的范围不在同一个区间。

那么float也是一样,它也是只能表示与int类型一样多的数字,只是float表示的数的范围是上面的那样而已。它能表示的数的范围很大,不要被这一点迷惑。

  • 可能我们觉得[-3.4 * 1038 , 3.4 * 1038] 这个区间很大,它的个数比int表示的个数多很多。 注意,这个想法是错误的!!!
  • float所表示的数不是连续的数,在[-3.4 * 1038 , 3.4 * 1038]这个区间,存在很多空洞,只有少部分是表示float的数。
  • float只是一种近似的表示法,不能作为精确数使用。

当然这一切,都是由于float的内存表示法的不同造成的。有兴趣的人可以研究一下,上述float的数的范围区间是如何来的。

2.1 float数不精确的编码案例

看下面的代码:

#include <stdio.h>int main()
{float f = 3.1415f;float fl = 123456789;printf("%0.10f\n", f);printf("%0.10f\n", fl);return 0;
}

编译运行结果为:

3.1414999962
123456792.0000000000

这个结果貌似与我们所期盼的结果有很大的出入。但是 这就是正常的结果,因为float表示的数是不精确的,像上面的数3.1415f ,float就无法表示这样的数,虽然它在区间[-3.4 * 1038 , 3.4 * 1038]中,但是float就是无法精确表示它,只能打印它最接近这个数的一个数。所以打印结果是另一个数。

  • 这也验证了上面关于说float无法精确表示一个数的说法

3 总结

  • 浮点数与整形数的内存表示法不同
  • 浮点数类型可表示的范围更大
  • 浮点类型是一种不精确的类型
  • 由于浮点数的内存表示法比较复杂,所以浮点数的运算速度很慢
  • float所表示的数不是连续的数

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

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

相关文章

【C语言进阶深度学习记录】四 C语言中的类型转换

今天学习C语言中的类型转换&#xff0c;包括隐式类型转换和显示类型转换 文章目录1 C语言中的数据类型转换1.1 强制类型转换1.11 强制类型转换代码分析1.&#xff12; 隐式类型转换1.21 隐式类型转换代码分析2 总结1 C语言中的数据类型转换 C语言中&#xff0c;可以进行数据类…

【C语言进阶深度学习记录】五 C语言中变量的属性

上一篇文章学习了C语言中的类型转换&#xff0c;点击链接查看&#xff1a;【C语言进阶深度学习记录】四 C语言中的类型转换. 文章目录1 C语言的变量属性1.1 auto关键字1.2 register关键字1.3 static 关键字1.4 代码案例分析1.5 extern 关键字1.6 代码案例分析2 总结1 C语言的变…

【C语言进阶深度学习记录】六 C语言中的分支语句

文章目录1 if 语句的分析1.1 if 语句中零值比较的注意点2 switch 语句的分析3 if 与switch语句使用代码案例分析4 if语句与switch语句的互换5 总结1 if 语句的分析 if 语句根据条件选择执行语句else 不能独立存在&#xff0c;且总是与距离它最近的if匹配else 语句可以连接其他…

【C语言进阶深度学习记录】七 C语言中的循环语句

文章目录1 循环语句分析1.1 do...while循环1.2 while循环1.3 for循环1.4 三种循环语句使用对比2 break和continue的区别3 总结1 循环语句分析 C语言中的循环语句主要有for循环&#xff0c;while循环和do…while循环。 循环语句的基本工作方式&#xff1a; 通过条件表达式判断…

【C语言进阶深度学习记录】八 C语言中void的分析

文章目录1 void的意义1.1 不存在void变量1.2 C标准1.3 void指针的意义1.4 通过void* 实现memset函数2 总结1 void的意义 void修饰函数的参数和返回值的时候&#xff1a; 如果函数没有返回值应该将其返回值声明为void如果函数没有参数&#xff0c;应该将函数的参数声明为void如…

【C语言进阶深度学习记录】九 C语言中const的详细分析

文章目录1 const的分析2 const本质的分析实验2.1 代码案例分析3 const修饰函数参数和返回值时的情况3.1 代码案例分析4 总结1 const的分析 不管是C语言还是C语言&#xff0c;const都是一个非常重要的关键字。今天这篇文章着重学习记录C语言中的const。C语言中稍有不同。 在C语…

【C语言进阶深度学习记录】十 C语言中:struct的柔性数组和union分析

本文并不讲C语言的基础 文章目录1 空struct的大小2 结构体与柔性数组2.1 柔性数组的使用方法2.2 柔性数组使用代码案例分析3 C语言中的union分析3.1 使用union判断系统大小端4 总结1 空struct的大小 C语言中的struct可以看成是变量的集合 如果一个struct里面什么都没有&#…

【C语言进阶深度学习记录】十一 C语言中enum,sizeof,typedef分析

文章目录1 enum 枚举类型的使用方法1.1 enum枚举类型的特殊意义1.2 代码分析&#xff1a;enum的使用2 sizeof 关键字的用法2.1 代码案例分析&#xff1a;sizeof的本质3 typedef的意义3.1 代码案例&#xff1a;typedef 的使用案例4 总结1 enum 枚举类型的使用方法 enum是C语言中…

【C语言进阶深度学习记录】十二 C语言中的:字符和字符串

文章目录1 C语言中的单引号和双引号1.1 双引号带来的BUG2 总结1 C语言中的单引号和双引号 C语言中的单引号用来表示字符字面量C语言中的双引号用来表示字符串字面量&#xff0c;存储于全局的只读存储区 注意上面的字符与字符串的区别 下面的程序片段是否合法&#xff1f; 上面…

PetShop的系统架构设计[转]

前言&#xff1a;PetShop是一个范例&#xff0c;微软用它来展示.Net企业系统开发的能力。业界有许多.Net与J2EE之争&#xff0c;许多数据是从微软的PetShop和Sun的PetStore而来。这种争论不可避免带有浓厚的商业色彩&#xff0c;对于我们开发人员而言&#xff0c;没有必要过多关…

【C语言进阶深度学习记录】十三 C语言中 ++和--操作符

学习交流加&#xff08;可免费帮忙下载CSDN资源&#xff09;&#xff1a;个人微信&#xff1a; liu1126137994学习交流资源分享qq群1&#xff08;已满&#xff09;&#xff1a; 962535112学习交流资源分享qq群2&#xff1a; 780902027 文章目录1 和--操作符的本质2 总结1 和–操…

【C语言进阶深度学习记录】十四 C语言中 三目运算符和逗号表达式

文章目录1 三目运算符1.1 三目运算符的返回类型的代码案例分析2 逗号表达式2.1 逗号表达式代码案例分析2.2 如何用一行代码实现 strlen函数3 总结1 三目运算符 三目运算符&#xff08;a?b:c&#xff09;可以作为逻辑运算的载体规则: 当a为真时&#xff0c;返回b&#xff0c;否…

POJ 1719 Shooting Contest

题目&#xff1a;http://poj.org/problem?id1719 要求每一行必须都被射到&#xff0c;每一列恰好一个格子被射到。 通过行r去求匹配数num 当r<c时&#xff0c;num是可以等于r的 这时候每一行都被射到过&#xff0c;可能存在列没有被射到&#xff0c;可以再该列中任意选一个…

【C语言进阶深度学习记录】十五 编译过程简介

文章目录1 初识编译器2 总结1 初识编译器 我们平时口中所说的编译器&#xff0c;是广泛的编译器。实际上&#xff0c;编译器包括了以下四个部分&#xff1a; 一个C代码被编译为可执行代码&#xff0c;包括以下几个过程&#xff1a; 下面就对上述的各个过程进行一个详细的说明&…

Android 各控件的使用 - 按钮(Button)

安卓按钮的使用 就不细说项目的设置什么的了&#xff0c;直接从添加Activity开始。 如图&#xff0c;src文件夹下添加一个ButtonActivity View Code packagecn.Kurodo;importcn.Kurodo.R.id;importandroid.app.Activity;importandroid.os.Bundle;importandroid.util.Log;impor…

【C语言进阶深度学习记录】十六 静态库与动态库的创建与使用

上一篇文章学习了编译的过程&#xff0c;点击链接查看&#xff1a;【C语言进阶深度学习记录】十五 编译过程简介&#xff0c;每一个C源文件编译后将会生成目标文件&#xff0c;那么这些目标文件&#xff0c;还需要链接起来&#xff0c;生成可执行文件。 文章目录1 链接的意义1.…

【C语言进阶深度学习记录】十七 宏定义的使用与分析

文章目录1 C语言中的宏定义1.1 定义宏常量1.2 宏定义表达式1.3 宏表达式与函数的对比1.4 宏表达式的作用域2 C语言中的内置宏3 宏定义的代码综合示例4 总结1 C语言中的宏定义 #define 是预处理器处理的单元实体之一#define 定义的宏&#xff0c;可以出现在程序的任意位置#defi…

【C语言进阶深度学习记录】十八 条件编译的使用与分析

文章目录1 基本概念1.1 代码分析1.2 通过命令行定义宏2 #include 的本质2.1 解决重复包含头文件的问题3 条件编译的应用4 总结1 基本概念 条件编译的行为类似于C语言中的if … else…条件编译是预编译指示指令&#xff0c;用于控制是否编译某段代码 比如下图的代码&#xff1…

【C语言进阶深度学习记录】十九 #pragma使用与分析

文章目录1 #pragma 概念简介1.1 #pragma message 的用法1.2 #pragma once 的用法1.3 #pragma pack 的用法1.31 struct占用的内存大小如何计算2 总结在学习 #pragma 之前 &#xff0c;我们首先要明白一点&#xff0c; #pragma 的实现&#xff0c;在不同的编译器之间是不同的&…