浮点数的运算为什么更慢

1. 浮点数的表示

 

m 是尾数, 为±d.dddddd   其中 第一位必须非0

b 是基数, 

下面,让我们回到一开始的问题:为什么0x00000009还原成浮点数,就成了0.000000?

首先,将0x00000009拆分,得到第一位符号位s=0,后面8位的指数E=00000000,最后23位的有效数字M=000 0000 0000 0000 0000 1001。

由于指数E全为0,所以符合上一节的第二种情况。因此,浮点数V就写成:

V=(-1)^0×0.00000000000000000001001×2^(-126)=1.001×2^(-146)

 

 

结论

通用规则:

  • 整数和(和ANDORXOR)与乘积花费的时间相同,除法(和取模)的速度慢三倍。
  • 浮点数的乘积比求和的乘积慢两倍,除法的乘积甚至更慢。
  • 在相同数据大小下,浮点运算始终比整数运算慢。
  • 越小越快。
  • 64位整数精度确实很慢。
  • 浮点数32位总和比64位快,但在乘积和除法上却不是。
  • 80和128位精度仅在绝对必要时才应使用,它们非常慢。

特别案例:

  • 在x86-64 AVX上,浮点乘积在64位数据上比在32位上更快
  • 在POWER8 AltiVec上,浮点乘积以各种精度达到求和的速度对8位,16位,32位或64位整数以相同的速度执行整数运算。
  • 在ARM1176上,按位整数运算符比加法运算要快。

 

 

示例代码:

#include <stdio.h>
#include <math.h>
#include <stdlib.h>
#include <cv/cv_tools.h>
#include <picture/cv_picture.h>
#include "libyuv.h"
using namespace cv;
using namespace std;
using namespace oop;


int main()
{
    const int  N= 10000;
    int sum = 0;
    float sumf = 0;

    float nf = 734.0f;
    int n = 734;
    timeInit;
    timeMark("int");
    for(int j=0;j!=100000;++j)
    {
        sum = 0;
        for (int i = 0; i != N; ++i) {
            sum += n;
        }
    }
    timeMark("float");
    for (int j = 0; j != 100000; ++j)
    {
        sumf = 0;
        for (int i = 0; i != N; ++i) {
            sumf += nf;
        }
    }
    timeMark(")");
    timePrint;

    printf("sum=%d\nsumf=%.2f\n",sum,sumf);
    getchar();
}

 

输出:

( int,float ) : 2107 ms
( float,) ) : 3951 ms

sum=7340000
sumf=7340000.00

 

Release:

( int,float ) : 0 ms
( float,) ) : 1814 ms

sum=7340000
sumf=7340000.00

 

实际上: Debug模式下, 两者时间差不了多少,两倍的关系

但是Release模式下, int 几乎很快就完成了!!  说明int型被优化得很好了,float型运算不容易被编译器优化!!!

 

 

我们在Release模式下, 优化设置为O2, 连接器设置为-优化以便于调试

查看int 乘法汇编指令:xmm0 表示128位的SSE寄存器,可见我们的代码都被优化为SSE指令了!!

 

 

 查看float 汇编代码:

感觉里面也有xmm 等SSE指令集, 至于为啥int型乘法比float乘法快很多,  还是有点搞不明白, 需要详细分析里面的汇编指令才能搞明白

网上关于这方面的资料太少了, 哎~~

 

 

我们再看看float 和 int乘法对图像进行处理的例子:

我们把BGR 3个通道分别乘以2 3 4 、 2.0f, 3.0f, 4.0f 然后输出, 这里我们不考虑溢出的问题, 仅仅对乘法的效率进行测试

设置为Release模式,O2

int main()
{
    cv::Mat src = imread("D:/pic/nlm.jpg");
    //cvtColor(src,src,CV_BGR2GRAY);
    resize(src,src,Size(3840*2,2160*2));
    cv::Mat dst0(src.size(), src.type());
    cv::Mat dst1(src.size(), src.type());

    int w = src.cols;
    int h = src.rows;
    int of3=0;
    timeInit;
    timeMark("int");
    for (int j = 0; j != h; ++j) {
        for (int i = 0; i != w; ++i) {
            //int of3 = (j*w + i) * 3;
            dst0.data[of3    ] = src.data[of3] * 2;
            dst0.data[of3 + 1] = src.data[of3 + 1] * 3;
            dst0.data[of3 + 2] = src.data[of3 + 2] * 4;
            of3+=3;
        }
    }
    timeMark("float");
    of3=0;
    for (int j = 0; j != h; ++j) {
        for (int i = 0; i != w; ++i) {
            //int of3 = (j*w + i)*3;
            dst1.data[of3] = src.data[of3] * 2.0f;
            dst1.data[of3+1] = src.data[of3+1] * 3.0f;
            dst1.data[of3+2] = src.data[of3+2] * 4.0f;
            of3 += 3;
        }
    }
    timeMark("end");
    timePrint;
    myShow(dst0);
    myShow(dst1);

    waitKey(0);
}

输出:

( int,float ) : 149 ms
( float,end ) : 173 ms

输出图像(分别为原图,dst0,dst1)(截取了一部分)

 可见,时间并差不了多少,但int还是要快一点!!

 

 这是我看到的另外一个帖子,里面讲的float乘法确实比较复杂 , 这可能是它比较慢的原因之一吧

https://blog.csdn.net/u014298090/article/details/21867187

 

总结一下: float运算更慢的原因:

1. float运算不容易被编译器优化

2. float运算本身就慢(但并不比int型运算慢多少,大约1.3-2倍的样子)

posted @ 2019-09-30 16:44  洛笔达  阅读(6303)  评论(0编辑  收藏  举报