吾爱破解 - 52pojie.cn

 找回密码
 注册[Register]

QQ登录

只需一步,快速开始

查看: 1908|回复: 15
收起左侧

[C&C++ 原创] 挑战全网最简洁快速平方根倒数算法讲解

[复制链接]
hans7 发表于 2023-11-2 15:06

一句话介绍

功能等同于函数(x) => 1 / Math.sqrt(x)源码传送门

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;
}

前置知识

牛顿迭代:本文用到的公式:y = 1 / sqrt(x) -> 1 / y^2 - x = 0代入得结果。

IEEE-754浮点数表示:一句话总结:正数的符号位肯定为0,x = (1 + M / 2^23) * 2^(E - 127)

推导过程

上述代码只有这几行

i  = * ( long * ) &y;
i  = 0x5f3759df - ( i >> 1 );
y  = * ( float * ) &i;

是不那么显然的。

考虑x ^ -0.5的对数:-0.5 * (log2(1 + M / 2^23) + (E - 127))log2(1 + M / 2^23)必定属于[0, 1),所以可以近似为M / 2^23。于是log2(x) ≈ (M + 2^23 * E - 2^23 * 127) / 2^23,其中M + 2^23 * E就是将浮点数解释为int后的值,记为blog2(x) ≈ b / 2^23 - 127。设结果A = log2(x ^ -0.5)的这一部分为a,则a / 2^23 - 127 = -0.5 * (b / 2^23 - 127),解得a = 381 * 2^22 - (b >> 1)

至此我们已经能够理解整个算法。但上述代码的magic number并不是381 * 2^22。别着急,我们马上讨论。

误差分析

从直觉来看需要将y = x稍微上移才能得到最优近似,所以我们设v = err(x) = log2(x + 1) - x, x ∈ [0, 1)。根据参考链接2,计算v常用的方式有:

  1. (max(err(v)) + min(err(v))) / 2err'(x) = 1 / (log(2) * (1 + x)) - 1,曲线先升后降,所以x = 1 / log(2) - 1得最大值,结果为0.0430357
  2. err(x)[0,1]上积分的平均。在wolframalpha中算积分Integrate[log2(1+x)-x,x]let g = x => (2*(x+1)*Math.log(x+1)-x*(x*Math.log(2)+2))/Math.log(4),代入得0.057304959

然而,这些都只是我个人的猜测。已知最终作者使用的magic number是0x5f3759df,我们来算下最终作者选用的v值。log2(1 + M / 2^23)的最优近似为M / 2^23 + v,于是log2(x) ≈ b / 2^23 - 127 - v => a / 2^23 - 127 - v = -0.5 * (b / 2^23 - 127 - v),得v = 0.0450465679

课后作业

实现double的版本。

#include <bits/stdc++.h>
// Copyright 2023 hans7

double q_rsqrt(double v) {
  int64_t tmp = *reinterpret_cast<int64_t*>(&v);
  tmp = 6910773628200026112LL - (tmp >> 1);
  double res = *reinterpret_cast<double*>(&tmp);
  res = res * (1.5 - v * 0.5 * res * res);
  return res;
}

int main(int, char**) {
  srand(time(nullptr));
  int v = rand();
  printf("%d %.10lf %.10lf\n", v, q_rsqrt(v), abs(q_rsqrt(v) - 1 / sqrt(v)));
  return 0;
}

参考资料

  1. http://www.matrix67.com/data/InvSqrt.pdf
  2. https://blog.csdn.net/Skywalker1111/article/details/116295522

免费评分

参与人数 1吾爱币 +10 热心值 +1 收起 理由
苏紫方璇 + 10 + 1 欢迎分析讨论交流,吾爱破解论坛有你更精彩!

查看全部评分

发帖前要善用论坛搜索功能,那里可能会有你要找的答案或者已经有人发布过相同内容了,请勿重复发帖。

dork 发表于 2023-11-2 16:21
发个python版的,原理一样:

class Solution():
    def mySqrt(self,num):
        t = num
        t = 0x5f3759df - (t >> 1)
        while not (t * t <= num and (t+1) * (t+1) > num):
            t = (num / t + t) / 2
        return t
天下客 发表于 2023-11-2 16:22
一脸懵地进来,一脸懵地出去。。。还可以这样
xsf277 发表于 2023-11-2 16:23
 楼主| hans7 发表于 2023-11-2 17:16
xsf277 发表于 2023-11-2 16:23
注释里有备注:WTF

copyright info是cpplint自带的规则qwq
 楼主| hans7 发表于 2023-11-2 17:25
dork 发表于 2023-11-2 16:21
发个python版的,原理一样:

class Solution():

这段代码有问题。输入mySqrt(3.5)会报错TypeError: unsupported operand type(s) for >>: 'float' and 'int'

用c/cpp以外的语言实现需要先把int转bytes再转float,算好后再从float转bytes再转int。

另外这个牛顿迭代式是算sqrt(x)的,不是算1/sqrt(x)的。

aqdy2021 发表于 2023-11-3 14:16
感谢分享,学习了。
a33521 发表于 2023-11-5 11:14
嗯 看看  学习一下
hotel 发表于 2023-11-5 11:28
学习了。
愚者—我 发表于 2023-11-6 14:42
xsf277 发表于 2023-11-2 16:23
注释里有备注:WTF

这段注释是Doom启示录的主人公之一, 卡马克写的. 具体的代码是他的同事所写.  而这段代码也是用于雷神之锤3这款游戏中.
而卡马克也是游戏引擎的开源先驱.
您需要登录后才可以回帖 登录 | 注册[Register]

本版积分规则

返回列表

RSS订阅|小黑屋|处罚记录|联系我们|吾爱破解 - LCG - LSG ( 京ICP备16042023号 | 京公网安备 11010502030087号 )

GMT+8, 2025-1-11 06:02

Powered by Discuz!

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表