C语言浮点函数中的modf和fmod详解

 更新时间:2022年02月18日 15:25:42   作者:嵌入式@hxydj  
这篇文章主要为大家详细介绍了C语言浮点函数中的modf和fmod,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助

modf函数可以提取出浮点数的整数部分和小数部分。fmod函数可以返回两个浮点数相除的余数。它们的函数原型如下:

  double __cdecl modf(double _X,double *_Y);
  double __cdecl fmod(double _X,double _Y);

这两个函数的功能看起来都挺简单的,但是为什么在C语言库中还要专门搞一个函数来计算呢?在使用这两个函数之前,首先看一个简单的浮点数相关的例子。

int main ()
{
	int i;
	float j=1.0;

	for(i=0; i<100; i++)
	{
		j+=0.1;
		printf("%f \n",j);
	}
	return(0);
}

给一个浮点数1.0每次累加0.1,总共累加100,感觉就是一个简单的加法运算,它的输出结果肯定也知道是什么了,下面来运行一下这个代码,输出结果如下:

image.png

看到结果的第一眼,是不是感觉计算机算错了?怎么算着算着结果还给跑偏了。其实计算机并没有算错,程序运行的结果也是正确的。这里使用一个在线浮点数转换工具验证一下。

首先看一下浮点数1.0在内存的存储情况。

image.png

再看一下浮点数2.8在内存的存储情况。

image.png

通过对比可以看出1.0的实际存储值是1,和真实值是一样的。2.8的实际存储值是2.7999999523162841796875,和真实值已经产生了一点误差。那么这个误差是什么引起的呢?这里就要从浮点数的存储原理说起了。

通在C语言中,一个浮点数占32位,其中第一位表示符号位,0表示正数,1表示负数。紧接着8位表示指数部分,也就是系统会将浮点数表示为指数计数法,这8位就是指数部分的值,但是这里的指数是以2为底的。剩下的23位表示的就是尾数,也就是有效数据。

由于小数部分都是通过 2^a+2^b+2^c.......2^n 这种指数的累加来实现的,而指数的累加不一定可以百分百的等于原值,它只可能无限的接近原值。所以对于浮点数的操作不是像整数那样简单,为了保证操作浮点数时的精度,所以C语言的库函数才给我们提供了这两种浮点数操作的函数,方便使用。

接下来看一下modf函数的用法。

#include <stdio.h>
#include <math.h>

int main ()
{
	double x, fractpart, intpart;

	x = 8.123456;
	fractpart = modf(x, &intpart);

	printf("整数部分= %lf\n", intpart);
	printf("小数部分 = %lf \n", fractpart);

	return(0);
}

modf函数有两个参数,第一个参数时待操作的浮点数,第二个参数用来存储浮点数的整数部分,它的返回值是浮点数的小数部分。

image.png

通过输出结果可以看到,浮点数的整数部分和小数部分精度都没有丢失。

再看一下fmod函数的用法。

#include <stdio.h>
#include <math.h>

int main ()
{
   float a, b;
   int c;
   a = 9.0;
   b = 3.7;
   c = 2;
      
   printf("%f / %f 的余数是 %lf\n", a, b, fmod(a,b));
   printf("%f / %d 的余数是 %lf\n", a, c, fmod(a,c));
   
	return(0);
}

fmod函数有两个参数,第一个参数是分子,第二个参数是分母,它返回第一个参数除以第二个参数的余数。

image.png

通过输出的结果可以看到,浮点数的精度没有丢失。 

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!

相关文章

  • 指针与const限定符的使用分析

    指针与const限定符的使用分析

    本篇文章对指针与const限定符的使用进行了分析介绍。需要的朋友参考下
    2013-05-05
  • C++示例讲解string容器

    C++示例讲解string容器

    c++相比c的一个好处就是实现了很多的容器和泛型算法,使得程序员的工作得到了很大的简化,本文重点给大家介绍C++string容器基本概念讲解,需要的朋友参考下吧
    2022-07-07
  • C++与namespace有关的两个编译错误的讲解

    C++与namespace有关的两个编译错误的讲解

    今天小编就为大家分享一篇关于C++与namespace有关的两个编译错误的讲解,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-04-04
  • Opencv光流运动物体追踪详解

    Opencv光流运动物体追踪详解

    这篇文章主要为大家详细介绍了Opencv光流运动物体追踪的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-01-01
  • C++实现LeetCode(73.矩阵赋零)

    C++实现LeetCode(73.矩阵赋零)

    这篇文章主要介绍了C++实现LeetCode(73.矩阵赋零),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • c语言中的局部跳转及全局跳转功能

    c语言中的局部跳转及全局跳转功能

    本文介绍了C语言中的goto语句,以及如何使用setjmp和longjmp实现跨函数的跳转,详细讲解了setjmp和longjmp的使用方法和注意事项,以及使用这种全局跳转后变量状态的不确定性,感兴趣的朋友一起看看吧
    2024-09-09
  • C语言实现简单三子棋游戏

    C语言实现简单三子棋游戏

    这篇文章主要为大家详细介绍了C语言实现简单三子棋游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-09-09
  • 有关C++中随机函数rand() 和srand() 的用法详解

    有关C++中随机函数rand() 和srand() 的用法详解

    下面小编就为大家带来一篇有关C++中随机函数rand() 和srand() 的用法详解。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-01-01
  • 利用Matlab实现绘制中秋山间秋月和皓月当空效果

    利用Matlab实现绘制中秋山间秋月和皓月当空效果

    中秋节还有三天就到了,中秋节啊,阖家团圆的日子。本文将利用Matlab绘制中秋山间秋月和皓月当空的动态效果,感兴趣的可以了解一下
    2022-09-09
  • 深入遍历二叉树的各种操作详解(非递归遍历)

    深入遍历二叉树的各种操作详解(非递归遍历)

    本篇文章是对遍历二叉树的各种操作进行了详细的分析介绍,需要的朋友参考下
    2013-05-05

最新评论