一文教你搞懂C语言的Q格式

用过DSP的应该都知道Q格式吧;

  • 1 前言

  • 2 Q数据的表示

    • 2.1 范围和精度

    • 2.2 推导

  • 3 Q数据的运算

    • 3.1 0x7FFF

    • 3.2 0x8000

    • 3.3 加法

    • 3.4 减法

    • 3.5 乘法

    • 3.6 除法

  • 4 常见Q格式的数据范围

  • 5 0x5f3759df

  • 6 总结

1 前言

Q格式是二进制的定点数格式,相对于浮点数,Q格式指定了相应的小数位数和整数位数,在没有浮点运算的平台上,可以更快地对浮点数据进行处理,以及应用在需要恒定分辨率的程序中(浮点数的精度是会变化的);
需要注意的是Q格式是概念上小数定点,通过选择常规的二进制数整数位数和小数位数,从而达到所需要的数值范围和精度,这里可能有点抽象,下面继续看介绍。

2 Q数据的表示

2.1 范围和精度

定点数通常表示为,其中m为整数个数,n为小数个数,其中最高位位符号位并且以二进制补码的形式存储;

  • 范围:
  • 精度:

无符号的用表示;

  • 范围:
  • 精度:

2.2 推导

无符号Q格式数据的推导这里以一个16位无符号整数为例,所能表示的最大数据的二进制形式如下图所示;

所以不难看出,的范围大小和精度;根据等比数列求和公式得到,整数域最大值如下:

小数域最大值如下:

因此的范围满足 ;

有符号Q格式数据的推导这里以一个16位有符号整数为例,所能表示的最大数据的二进制形式如下图所示;

所以不难求出,的范围大小和精度;根据等比数列求和公式得到,整数域最大值如下:

小数域最大值如下:

因此最大能表示的数为:;

所能表示的最小数据的二进制形式如下图所示;

可以从图中看到,该数表示为;

补充一下:负数在计算机中是补码的形式存在的,补码=反码+1,符号位为1则表示为负数;
那么-4该如何表示呢?
8 bit数据为例,如下所示;
原码:0B 0000 100
反码:0B 1111 011
补码:0B 1111 100

综上,可以得到有符号的范围是:

3 Q数据的运算

3.1 0x7FFF

最大数的十六进制为0x7FFF,如下图所示;

3.2 0x8000

最小数的十六进制为0X8000,如下图所示;

上述这两种情况,下面都会用到。

3.3 加法

加法和减法需要两个Q格式的数据定标相同,即和满足以下条件;

int16_t q_add(int16_t a, int16_t b)
{
    return a + b;
}

上面的程序其实并不安全,在一般的DSP芯片具有防止溢出的指令,但是通常需要做一下溢出检测,具体如下所示;

//https://great.blog.csdn.net/int16_t q_add_sat(int16_t a, int16_t b){    int16_t result;    int32_t tmp;

    tmp = (int32_t)a + (int32_t)b;    if (tmp > 0x7FFF)        tmp = 0x7FFF;    if (tmp < -1 * 0x8000)        tmp = -1 * 0x8000;    result = (int16_t)tmp;

    return result;}

3.4 减法

类似于加法的操作,需要相同定标的两个Q格式数进行相减,但是不会存在溢出的情况;

//https://great.blog.csdn.net/
int16_t q_sub(int16_t a, int16_t b)
{
    return a - b;
}

3.5 乘法

乘法同样需要考虑溢出的问题,这里通过sat16函数,对溢出做了处理;

//https://great.blog.csdn.net/// precomputed value:#define K   (1 << (Q - 1))

// saturate to range of int16_tint16_t sat16(int32_t x){    if (x > 0x7FFF) return 0x7FFF;    else if (x < -0x8000) return -0x8000;    else return (int16_t)x;}

int16_t q_mul(int16_t a, int16_t b){    int16_t result;    int32_t temp;

    temp = (int32_t)a * (int32_t)b; // result type is operand's type    // Rounding; mid values are rounded up    temp += K;    // Correct by dividing by base and saturate result    result = sat16(temp >> Q);

    return result;}

3.6 除法

//https://great.blog.csdn.net/
int16_t q_div(int16_t a, int16_t b)
{
    /* pre-multiply by the base (Upscale to Q16 so that the result will be in Q8 format) */
    int32_t temp = (int32_t)a << Q;
    /* Rounding: mid values are rounded up (down for negative values). */
    /* OR compare most significant bits i.e. if (((temp >> 31) & 1) == ((b >> 15) & 1)) */
    if ((temp >= 0 && b >= 0) || (temp < 0 && b < 0)) {   
        temp += b / 2;    /* OR shift 1 bit i.e. temp += (b >> 1); */
    } else {
        temp -= b / 2;    /* OR shift 1 bit i.e. temp -= (b >> 1); */
    }
    return (int16_t)(temp / b);
}

4 常见Q格式的数据范围

定点数和浮点数转换的关系满足以下公式:

其中为,m表示整数位数,n表示小数位数;

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

int main(){    // 0111 1111 1111 1111    int16_t q_max = 32767; // 0x7FFF    // 1000 0000 0000 0000    int16_t q_min = -32768; // 0x8000    float f_max = 0;    float f_min = 0;    printf('\r\n');    for (int8_t i = 15; i>=0; i--) {        f_max = (float)q_max / pow(2,i);        f_min = (float)q_min / pow(2,i);

        printf('\t| Q %d | Q %d.%d| %f | %f |\r\n',               i,(15-i),i,f_max,f_min);    }

    return 0;}

运行得到结果如下所示;

Q Qmn Max Min
Q 15 Q 0.15 0.999969 -1.000000
Q 14 Q 1.14 1.999939 -2.000000
Q 13 Q 2.13 3.999878 -4.000000
Q 12 Q 3.12 7.999756 -8.000000
Q 11 Q 4.11 15.999512 -16.000000
Q 10 Q 5.10 31.999023 -32.000000
Q 9 Q 6.9 63.998047 -64.000000
Q 8 Q 7.8 127.996094 -128.000000
Q 7 Q 8.7 255.992188 -256.000000
Q 6 Q 9.6 511.984375 -512.000000
Q 5 Q 10.5 1023.968750 -1024.000000
Q 4 Q 11.4 2047.937500 -2048.000000
Q 3 Q 12.3 4095.875000 -4096.000000
Q 2 Q 13.2 8191.750000 -8192.000000
Q 1 Q 14.1 16383.500000 -16384.000000
Q 0 Q 15.0 32767.000000 -32768.000000

5 0x5f3759df

Q格式虽然十分抽象,但是且看看这个数字0x5f3759df,感觉和Q格式有某种联系,它是雷神之锤3中的一个算法的魔数,毕竟游戏引擎需要充分考虑到效率,具体的由来可以看一下论文《Fast Inverse Square Root》,下面是源码中剥出来的快速平方根算法;

float Q_rsqrt( float number )
{
 long i;
 float x2, y;
 const float threehalfs = 1.5F;

x2 = number * 0.5F;
 y   = number;
 i   = * ( long * ) &y;   // evil floating point bit level hacking
 i   = 0x5f3759df - ( i >> 1 ); // what the fuck?
 y   = * ( float * ) &i;
 y   = y * ( threehalfs - ( x2 * y * y ) ); // 1st iteration
 // y   = y * ( threehalfs - ( x2 * y * y ) ); // 2nd iteration, this can be removed

#ifndef Q3_VM
 #ifdef __linux__
   assert( !isnan(y) ); // bk010122 - FPE?
 #endif
 #endif
 return y;
}  

6 总结

本文介绍了Q格式的表示方式以及相应的运算,另外需要注意在Q格式运算的时候,两者定标必须相同,对于数据的溢出检测也要做相应的处理。


作者能力有限,文中难免有错误和纰漏之处,请大佬们不吝赐教 创作不易,如果本文帮到了您;请帮忙点个赞 👍👍👍;

(0)

相关推荐

  • c 如何求任意多边形的面积

    #include <iostream> #include <vector> using namespace std; struct Point//定义坐标结构体 { float ...

  • 一文教你搞懂装修全流程(工序 采购)!

    长图镇楼.(建议点开原图再保存) 不方便看图的,后面有文字版,全是干货 # 01 装修准备阶段 第一步,找设计师 设计师分两类: 1.装修公司的设计师.设计费大多不高,有的甚至"免费设计&q ...

  • 搞懂程序语言与计算机硬件的关系

    IT行业程序开发使用到了很多种高级语言比如Java,C,Python,JavaScript,也有个别领域用到了低级语言即汇编语言,这些都是程序语言,程序语言就是人类能够理解的语言,计算机硬件也有语言即 ...

  • 摄影:先构图还是先对焦?一文教你搞懂对焦构图的先后问题

    在我们拍摄照片的过程中,必然存在构图.对焦两个操作,关键在于是先构图呢?还是先对焦呢?今天我们来好好说道这个常见的问题,把先构图后对焦还是先对焦后构图搞明白. 一.先对焦.后构图 很多人对于对焦.构图 ...

  • 一文搞懂C语言回调函数

    什么是回调函数 我们先来看看百度百科是如何定义回调函数的: 回调函数就是一个通过函数指针调用的函数.如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这 ...

  • 有人喝了一辈子茶都没搞懂这个问题?一文教你搞懂六大茶类

    最近越来越多的茶友都找到了我们的小组织,每天在茶友群里山南海北的聊天,聊茶,也聊生活,但在聊茶的过程中,却非常遗憾的发现一个问题:很多新手茶友对我国传统的六大茶类了解非常有限,有人分不清红茶和乌龙茶, ...

  • 搞懂C语言指针,看这篇就够了!

    说到指针,估计还是有很多小伙伴都还是云里雾里的,有点"知其然,而不知其所以然".但是,不得不说,学了指针,C语言才能算是入门了.指针是C语言的「精华」,可以说,对对指针的掌握程度, ...

  • 搞懂C语言函数指针

    原文地址:https://www.yanbinghu.com/2019/01/03/3593.html 前言 函数指针是什么?如何使用函数指针?函数指针到底有什么大用?本文将一一介绍. 如何理解函数指 ...

  • 爱丽舍艺术涂料一文教你搞懂家装全流程

    装修是一件系统而繁杂的工作,如果你没有新房装修流程知识,肯定会手忙脚乱,这里爱丽舍小编为你详细介绍新房装修的20个步骤流程,让你少走弯路少花钱. 一.前期设计 在前期设计中,必须要做一件事,就是对自己 ...

  • C语言丨还没搞懂逻辑运算符?一篇文章教你区分“真假”!

    这两天有遇到一些小伙伴,对于逻辑运算符号都不认识,不知道它代表什么意思,今天就来简单讲讲逻辑运算符.   逻辑运算符 关系运算符只是测试左右两个值之间的关系(把它们相互比较),逻辑运算符(logica ...