【C语言深度解剖】float变量在内存中存储原理&&指针变量与“零值”比较

这篇具有很好参考价值的文章主要介绍了【C语言深度解剖】float变量在内存中存储原理&&指针变量与“零值”比较。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

大家好好我是沐曦希💕

float中0是一个很小的数,C语言零基础学习,c语言,机器学习,算法


float中0是一个很小的数,C语言零基础学习,c语言,机器学习,算法

🎆float 变量

💥浮点数在内存中的存储原理

根据国际标准IEEE(电气和电子工程协会) 754,任意一个二进制浮点数V可以表示成下面的形式:

(-1)^S * M * 2^E
(-1)^S表示符号位,当S=0,V为正数;当S=1,V为负数。
M表示有效数字,大于等于1,小于2。
2^E表示指数位。

例如:
十进制中的5.0转换成二进制是101.0,相当于1.01*2^2
(二进制中0.1表示1*2^(-1); 0.11表示1*2^(-1)+1*2^(-2))
那么,按照上面V的格式,可以得出S=0,M=1.01,E=2。
十进制的-5.0,写成二进制是 -101.0 ,相当于 -1.01×2^2 。那么,S=1,M=1.01,E=2。
V=9.5f=1001.1=(-1)^0*1.0011*2^3
相当于S=0;M=1.0011;E=3。

🔭IEEE 754规定

对于32位的浮点数,最高的1位是符号位s,接着的8位是指数E,剩下的23位为有效数字M。
单精度浮点数存储模型:
float中0是一个很小的数,C语言零基础学习,c语言,机器学习,算法
对于64位的浮点数,最高的1位是符号位S,接着的11位是指数E,剩下的52位为有效数字M。
双精度浮点数存储模型:
float中0是一个很小的数,C语言零基础学习,c语言,机器学习,算法
IEEE 754对有效数字M和指数E,还有一些特别规定。

🎠有效数字M的规定

前面说过, 1≤M<2 ,即M可以写成 1.xxxxxx 的形式,其中xxxxxx表示小数部分。

IEEE 754规定,在计算机内部保存M时,默认这个数的第一位总是1,因此可以被舍去,只保存后面的xxxxxx部分。比如保存1.01的时候,只保存01,等到读取的时候,再把第一位的1加上去。这样做的目的,是节省1位有效数字。以32位浮点数为例,留给M只有23位,将第一位的1舍去以后,等于可以保存24位有效数字

🚗指数E的规定

🎉指数E为一个无符号整数

这意味着,如果E为8位,它的取值范围为0~255;如果E为11位,它的取值范围为0~2047。存入内存时E的真实值必须再加上一个中间数,对于8位的E,这个中间数是127;对于11位的E,这个中间数是1023

例如: V=0.5f;
(-1)^0*1.0*2^(-1) 保存成32位浮点数时,必须保存成E+127–>126
保存成64位浮点数时,必须保存成E+1023–>1022 即float–>E(真实值)+127(中间值)–>存储值
即double–>E(真实值)+1023(中间值)–>存储值

然后,指数E从内存中取出还可以再分成三种情况:

💥E不全为0或不全为1

浮点数就采用下面的规则表示,即指数E的计算值减去127(或1023),得到真实值,再将有效数字M前加上第一位的1。

💥E全为0

当浮点数的指数E等于1-127(或者1-1023)即为真实值,
有效数字M不再加上第一位的1,而是还原为0.xxxxxx的小数。这样做是为了表示±0,以及接近于0的很小的数字。

💥E全为1

当有效数字M全为0,表示±无穷大(正负取决于符号位s);

🎇float 变量与"零值"进行比较

浮点数在内存中存储,并不想我们想的,是完整存储的,在十进制转化成为二进制,是有可能有精度损失的。
注意这里的损失,不是一味的减少了,还有可能增多。浮点数本身存储的时候,在计算不尽的时候,会“四舍五入”或者其他策略

//code1
#include <stdio.h>
int main()
{
	double x = 3.6;
	printf("%.50f\n", x);
	return 0;
}

float中0是一个很小的数,C语言零基础学习,c语言,机器学习,算法

//code2
#include <stdio.h>
int main()
{
	double x = 1.0;
	double y = 0.1;
	printf("%.50f\n", x - 0.9);
	printf("%.50f\n", y);
	if ((x - 0.9) == y) 
	{
		printf("you can see me!\n");
	}
	else
	{
		printf("oops\n");
	}
	return 0;
}

float中0是一个很小的数,C语言零基础学习,c语言,机器学习,算法

结论:因为精度损失问题,两个浮点数,绝对不能使用==进行相等比较.
浮点数本身有精度损失,进而导致各种结果可能有细微差别。

浮点数比较应该进行范围精度比较
伪代码一:

if((x-y) > -精度 && (x-y) < 精度){
//TODO
}

伪代码-简洁版

if(fabs(x-y) < 精度){ //fabs是浮点数求绝对值
//TODO
}

精度可以直接定义一个(宏定义)或者使用系统精度。

🎡宏定义精度

#define EPS 0.0000000000000001
#include<stdio.h>
#include<math.h>
int main()
{
	double x = 1.0;
	double y = 0.1;
	if (fabs((x - 0.9) - y) < EPS)
	{
		printf("x==y");
	}
	else
	{
		printf("x!=y");
	}
	return 0;
}

输出结果:
float中0是一个很小的数,C语言零基础学习,c语言,机器学习,算法

🎡系统精度

#include<float.h> //使用下面两个精度,需要包含该头文件
DBL_EPSILON //double 最小精度
FLT_EPSILON //float 最小精度
#include <stdio.h>
#include <math.h> //必须包含math.h,要不然无法使用fabs
#include <float.h> //必须包含,要不然无法使用系统精度
int main()
{
	double x = 1.0;
	double y = 0.1;
	printf("%.50f\n", x - 0.9);
	printf("%.50f\n", y);
	if (fabs((x - 0.9) - y) < DBL_EPSILON) 
	{ //原始数据是浮点数,我们就用DBL_EPSILON
		printf("x==y\n");
	}
	else 
	{
		printf("x!=y\n");
	}
	return 0;
}

float中0是一个很小的数,C语言零基础学习,c语言,机器学习,算法

两个精度定义
#define DBL_EPSILON 2.2204460492503131e-016 /* smallest such that 1.0+DBL_EPSILON !=1.0 */
#define FLT_EPSILON 1.192092896e-07F /* smallest such that 1.0+FLT_EPSILON !=1.0 */
XXX_EPSILON是最小误差,是:XXX_EPSILON+n不等于n的最小的正数。
EPSILON这个单词翻译过来是’ε’的意思,数学上,就是极小的正数。

🧨x与"零值"比较

#include <stdio.h>
#include <math.h>
#include <float.h>
int main()
{
	double x = 0.00000000000000000000001;
	//if (fabs(x-0.0) < DBL_EPSILON){ //写法1
	//if (fabs(x) < DBL_EPSILON){ //写法2
	if (x > -DBL_EPSILON && x < DBL_EPSILON) 
	{ 
		printf("x==y\n");
	}
	else 
	{
		printf("x!=y\n");
	}
	return 0;
}

float中0是一个很小的数,C语言零基础学习,c语言,机器学习,算法

x > -DBL_EPSILON && x < DBL_EPSILON: 为何不是>= && <= 呢?
XXX_EPSILON是最小误差,是:
XXX_EPSILON+n不等于n的最小的正数。
XXX_EPSILON+n不等于n的最小的正数: 有很多数字+n都可以不等于n,但是XXX_EPSILON是最小的,但是XXX_EPSILON依旧是引起不等的一员。
换句话说:fabs(x) <= DBL_EPSILON(确认x是否是0的逻辑),如果=,就说明x本身,已经能够引起其他和他±的数据本身的变化了,这个不符合0的概念。

🎇fabs库函数

float中0是一个很小的数,C语言零基础学习,c语言,机器学习,算法
fabs是用来计算浮点参数的绝对值的库函数,头文件是math.h。

Example
/* ABS.C: This program computes and displays
 * the absolute values of several numbers.
 */
#include  <stdio.h>
#include  <math.h>
#include  <stdlib.h>
void main( void )
{
   int    ix = -4, iy;
   long   lx = -41567L, ly;
   double dx = -3.141593, dy;
   iy = abs( ix );
   printf( "The absolute value of %d is %d\n", ix, iy);
   ly = labs( lx );
   printf( "The absolute value of %ld is %ld\n", lx, ly);
   dy = fabs( dx );
   printf( "The absolute value of %f is %f\n", dx, dy );
}

Output
The absolute value of -4 is 4
The absolute value of -41567 is 41567
The absolute value of -3.141593 is 3.141593

🎎指针变量与“零值”进行比较

#include<stdio.h>
int main()
{
	printf("%d\n", 0);//0
	printf("%d\n", '0');//字符0,ASCII码值48
	printf("%d\n", '\0');//字符串结束标'\0',ASCII码值0
	printf("%d\n", NULL);//0
	//#define NULL ((void *)0)
	return 0;
}

float中0是一个很小的数,C语言零基础学习,c语言,机器学习,算法
真实转化
字符串A为“123456”转换成int类型,此时字符串A的字节数为7,而int类型为4个字节,要进行真实转化,需要我们编写算法,使用相关库函数来进行转化。

强制类型转化
不改变内存中的数据,只改变对应的类型。
例如:

float a = 3.14;//3.14类型为double,存储在float类型中需要经过强制类型转化
//float a =(float)3.14;
int* p = NULL;//定义指针一定要同时初始化,否则为野指针
//if(p==0); if(p!=0);//不推荐
//if(p);  if(!p);//不推荐
if(NULL==p);  if(NULL!=p);//推荐

🧵else 到底与哪个if配对呢?

//code1
#include<stdio.h>
int main()
{
	int x = 0;
	int y = 1;
	if (10 == x)
		if (11 == y)
			printf("hello\n");
	else
			printf("world!\n");
	return 0;
}

无打印结果。

//code2
#include<stdio.h>
int main()
{
	int x = 0;
	int y = 1;
	if (10 == x)
	{
		if (11 == y)
		{
			printf("hello\n");
		}
	}
	else
	{
		printf("world!\n");
	}
	return 0;
}

打印结果:world!

结论:else 匹配if采取就近原则

💫写在最后

友友们觉得不错的可以给个关注,点赞或者收藏哦!😘感谢各位友友们的支持。

你的❤️点赞是我创作的动力的源泉
你的✨收藏是我奋斗的方向
你的🙌关注是对我最大的支持
你的✏️评论是我前进的明灯
创作不易,希望大佬你支持一下小沐吧😘文章来源地址https://www.toymoban.com/news/detail-620961.html

到了这里,关于【C语言深度解剖】float变量在内存中存储原理&&指针变量与“零值”比较的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • C语言—深度剖析数据在内存中的存储

    内置类型(C语言本身就具有的类型): 类型的意义: 1.使用这个类型开辟内存空间的大小(大小决定了使用的范围) 2.如何看待内存空间的视角 整形家族: 注:字符类型底层存储的是这个字符的ASCII值,ASCII值是个整数 浮点数家族: 构造类型(自定义类型): 数组类型 注

    2024年02月03日
    浏览(30)
  • 【C语言】深度剖析数据在内存中的存储

    目录 一、数据类型详细介绍  1.1、类型的基本归类 二、整形在内存中的存储 2.1、大小端字节序介绍及判断 三、浮点型在内存中的存储 3.1、浮点数存储规则 类型的意义: 1、使用这个类型开辟内存空间的大小(大小决定了使用范围)。 2、如何看待内存空间的视角。 char   

    2024年02月07日
    浏览(36)
  • 《C语言深度解剖》.pdf

    🐇 🔥博客主页: 云曦 📋系列专栏: 深入理解C语言 💨吾生也有涯,而知也无涯 💛 感谢大家👍点赞 😋关注📝评论 C语言深度解剖.pdf 提取码:yunx

    2024年02月13日
    浏览(29)
  • 温故c语言——深度剖析数据在内存中的存储

    数据类型详细介绍 整形在内存中的存储:原码、反码、补码 大小端字节序介绍及判断 浮点型在内存中的存储解析 基本内置数据类型有: 类型的意义: 使用这个类型开辟内存空间的大小(大小决定了使用范围)。 如何看待内存空间的视角。 1.1 类型的基本归类: 整形家族:

    2023年04月22日
    浏览(52)
  • 进阶C语言-深度剖析数据在内存中的存储

    📝通过前面的学习,我们已经学习了基本的内置类型: 🔎 类型的意义 : 使用这个类型开辟内存空间的大小。 如何看待内存空间的视角。 🔭 整形家族 : 🔭 浮点数家族 : 🔭 构造类型(自定义类型) : 🔭 指针类型 : 🔭 空类型 : void 表示空类型(无类型) 通常用于

    2024年02月09日
    浏览(26)
  • 【C语言深度剖析——第四节(关键字4)】《C语言深度解剖》+蛋哥分析+个人理解

    追求本质,不断进步 本文由@睡觉待开机原创,转载请注明出处。 本内容在csdn网站首发 欢迎各位点赞—评论—收藏 如果存在不足之处请评论留言,共同进步! 前言: 本节博客继续前篇内容进行续写,我们着重探求有符号与无符号数的问题,探求 整形在内存中的存储 这一课

    2024年01月20日
    浏览(34)
  • 【C语言深度剖析——第三节(关键字3)】《C语言深度解剖》+蛋哥分析+个人理解

    本文由@睡觉待开机原创,未经允许不得转载。 本内容在csdn网站首发 欢迎各位点赞—评论—收藏 如果存在不足之处请评论留言,共同进步! 前言: 本期我们继续探讨关于C深度解剖这本书相关内容,继上一篇博客,本篇博客来围绕“数据类型”、“原反补”有关话题进行探

    2024年01月21日
    浏览(27)
  • 《C语言深度解剖》(4):深入理解一维数组和二维数组

    🤡博客主页:醉竺 🥰 本文专栏: 《C语言深度解剖》 😻 欢迎关注: 感谢大家的点赞评论+关注,祝您学有所成! ✨✨💜💛想要学习更多 数据结构与算法 点击专栏链接查看💛💜✨✨   数组是一组相同类型元素的集合。 数组的创建方式:  数组创建的实例:  注:数组

    2024年04月09日
    浏览(74)
  • 《C和指针》笔记12: 存储类型(自动变量、静态变量和寄存器变量)

    在 代码块内部声明的变量 的缺省存储类型是自动的(automatic),也就是说它存储于堆栈中,称为自动(auto)变量 。auto就是用于修饰这种存储类型的,但它极少使用,因为代码块中的变量在缺省情况下就是自动变量。 在程序执行到声明自动变量的代码块时,自动变量才被创

    2024年02月10日
    浏览(29)
  • C语言基本语句(变量类型int、 float、 double、 char,函数scanf、printf、putchar()、getchar() )

    1. int, float, double, char ①整型int(对应%d)  int a,b;  scanf(\\\"%d,%d\\\",a,b); printf (\\\"%d\\\",a); printf(\\\"我今天吃了%d个苹果,在黑板上写下整数%d,这很有趣。\\\",a,b); //printf(\\\"……\\\",变量名)中,“……”部分内容比较自由,可随便发挥,但必须包括%d,几个变量名就对应几个%d ②单精度型浮点数

    2024年02月08日
    浏览(34)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包