Roiban1344's Logbook

第11日目 - sin

LeetCode

登録して1問だけ解いた.競プロみたいに言語間での公平性を保証する必要がないため,標準入力のパースは不要で関数を定義するというスタイルが新鮮.

LeetCode - The World’s Leading Online Programming Learning Platform

これはどんどん解いていきたい.

sin1の近似値

MD5のこれに関して.

This step uses a 64-element table T[1 … 64] constructed from the sine function. Let T[i] denote the i-th element of the table, which is equal to the integer part of 4294967296 times abs(sin(i)), where i is in radians. The elements of the table are given in the appendix.

区間 [0,a]R[0,a]\in\mathbb{R} で定義されたCn+1C^{n+1}級関数f(x)f(x)に対して,

f(x)=k=0nf(0)k!xk+f(c)(n+1)!xn+1f(x) = \sum_{k=0}^n \frac{f(0)}{k!}x^k + \frac{f(c)}{(n+1)!}x^{n+1}

を満たすc(0,a)c\in(0,a)が存在する.f(x)=sinxf(x)=\sin xnn2n+12n+1 に読み替えて x>0x > 0 に対してこれを適用すると,

sinx=k=0n(1)kx2k+1(2k+1)!+(1)n+1sincx2n+3(2n+3)!\sin x = \sum_{k=0}^n \frac{(-1)^kx^{2k+1}}{(2k+1)!} + \frac{(-1)^{n+1}\sin c\, x^{2n+3}}{(2n+3)!}

を満たす c(0,x)c\in (0,x) が存在して sinx0\|\sin x\|\leq 0 だから,

sinxk=0n(1)kx2k+1(2k+1)!x2n+3(2n+3)!\left|\sin x-\sum_{k=0}^n \frac{(-1)^kx^{2k+1}}{(2k+1)!}\right| \leq \frac{x^{2n+3}}{(2n+3)!}

という,sin\sinのテイラー級数の有限項での打ち切りの誤差の厳密な評価が与えられる.同様に,

cosxk=0n(1)kx2k(2k)!x2n+2(2n+2)!.\left|\cos x-\sum_{k=0}^n \frac{(-1)^kx^{2k}}{(2k)!}\right| \leq \frac{x^{2n+2}}{(2n+2)!}.

OK.

さてx=1x=1とする.だいたいπ/3\pi/3だから

sin13/2=0.866,cos11/2=0.5\begin{align} \sin 1&\simeq \sqrt{3}/2=0.866\cdots,\\ \cos 1&\simeq 1/2=0.5 \end{align}

になるはずである.誤差項がxxに関して5次になる場合を使うともっと良い精度かつ厳密な評価が得られて,

sin1561120,cos113241120\begin{align} \left|\sin 1 - \frac{5}{6}\right| &\leq \frac{1}{120},\\ \left|\cos 1 - \frac{13}{24}\right| &\leq \frac{1}{120} \end{align}

から

sin1[100120,101120],cos1[65120,66120]\begin{align} \sin 1\in \left[\frac{100}{120},\, \frac{101}{120}\right],\\ \cos 1\in \left[\frac{65}{120},\, \frac{66}{120}\right] \end{align}

が分かる.

一方,x=1/2x=1/2の場合,

sin12[2348,2348+112025],cos12[337384,337384+112025]\begin{align} \sin \frac{1}{2} \in \left[\frac{23}{48},\, \frac{23}{48}+\frac{1}{120\cdot 2^5}\right],\\ \cos \frac{1}{2} \in \left[\frac{337}{384},\, \frac{337}{384}+\frac{1}{120\cdot 2^5}\right] \end{align}

倍角公式から,

sin1=2sin12cos12[77519216,62060117372800],cos1=cos212sin212[8852911638400,8864491638400].\begin{align} \sin 1 = 2\sin \frac{1}{2}\cos\frac{1}{2} &\in\left[ \frac{7751}{9216},\, \frac{6206011}{7372800} \right],\\ \cos 1 = \cos^2\frac{1}{2}-\sin^2\frac{1}{2} &\in\left[ \frac{885291}{1638400}, \frac{886449}{1638400} \right]. \end{align}

あえて厳密に有理数として計算する.すると区間の幅はどちらも579819200\frac{579}{819200}になる.一致することはごく単純な計算から分かるが,一瞬意表を突かれた.

この逆数が1414.81414.8\cdotsなので,x=1x=1のテイラー展開を使う場合の誤差項が66次の場合の約半分になる.あれ,それほど劇的に良いわけでもない…….

というのは次数が低いからで,次数が高いほどx=1/2x=1/2のほうの収束の速さが効いてくる.

0に十分近いxx

sinx[an1,an1+xnn!],cosx[bn1,bn1+xnn!],sinx2[an1,an1+xn2nn!],cosx2[bn1,bn1+xn2nn!]\begin{aligned} \sin x &\in \left[a_{n-1}, a_{n-1}+\frac{x^n}{n!}\right],& \cos x &\in \left[b_{n-1}, b_{n-1}+\frac{x^n}{n!}\right],\\ \sin \frac{x}{2} &\in \left[a'_{n-1}, a'_{n-1}+\frac{x^n}{2^nn!}\right],& \cos \frac{x}{2} &\in \left[b'_{n-1}, b'_{n-1}+\frac{x^n}{2^nn!}\right] \end{aligned}

の場合に後者の2倍角から計算したときの区間幅は

2(an1+bn1+xn2nn!)xn2nn!<xn2n2n!2\left(a'_{n-1}+b'_{n-1}+\frac{x^n}{2^n n!}\right)\frac{x^n}{2^n n!} < \frac{x^n}{2^{n-2} n!}

だから,冪で区間幅の縮小は速くなる.

しかし,十分大きなnnでは倍角を計算するより次数を上げるほうが区間幅は狭まるのか.ならsin(1/2)\sin (1/2)からsin1\sin 1を計算するよりsin1\sin 1を計算する方が速い?

実際にやれよという話で,やりたい.が準備が足りない.

一旦232sin12^{32} \sin 1の整数部分を求めるところまでやろう.

12!=479001600<232=4294967296<13!=622702080012! = 479001600 < 2^{32} = 4294967296 < 13 != 6227020800

なので,1313次までの誤差項があれば精度は足りる.

a12=113!+15!17!+19!111!=3358882939916800a_{12} = 1 - \frac{1}{3!} + \frac{1}{5!} - \frac{1}{7!} + \frac{1}{9!} - \frac{1}{11!} = \frac{33588829}{39916800} sin1[a12, a12+113!]\sin 1 \in \left[ a_{12},\ a_{12} + \frac{1}{13!} \right]

から,

232sin1[3614090359.,3614090360.].\lfloor 2^{32}\sin 1 \rfloor \in \left[ 3614090359.\ldots, 3614090360.\ldots \right].

「精度は足りる」←嘘だった.

a13:=113!+15!17!+19!111!+113!=209594293249080832a_{13} := 1 - \frac{1}{3!} + \frac{1}{5!} - \frac{1}{7!} + \frac{1}{9!} - \frac{1}{11!} + \frac{1}{13!} =\frac{209594293}{249080832}

まで精度を上げると,

232a13=232(a13+114!)=3614090360\left\lfloor 2^{32}a_{13} \right\rfloor =\left\lfloor 2^{32}\left(a_{13}+\frac{1}{14!}\right)\right\rfloor =3614090360

から,232sin12^{32}\sin 1 の整数部分は 36140903603614090360 だと結論付けられる.Hexではd76aa478.

RFC1321中のMD5のCによる実装(p.12)を見ると,この値がしっかり現れている.

  /* Round 1 */
  FF (a, b, c, d, x[ 0], S11, 0xd76aa478); /* 1 */

232sin642^{32}\sin 64を何の工夫もなしにテイラー級数のみから求めるには,まず誤差項が11未満になるために171171次まで要求される:

64171171!<1<64170170!.\frac{64^{171}}{171!} < 1 < \frac{64^{170}}{170!}.

この17117164e17464\,e\simeq 174から来ている(スターリングの公式).232sin642^{32}\sin 64の近似値の区間幅が1未満になるにはもう少し必要で,192192次になる.これをまともに有理数のまま計算すると桁数が爆発する.

……とはいってもせいぜい100桁オーダーなので数式処理システムは難なくやってのけてしまうが.ただ分母にどんどん値の評価に本質的ではない素因数が蓄積してしまうのが気持ち良くない.

精度保証付き数値計算

精度保証付き数値計算における四則演算の基本は数を「区間」として操作することで,区間演算という.区間演算の遂行には丸めの方向を厳密に取り扱う必要があるが,浮動小数点数は丸めの方向が実装依存になっているためそのままでは実現できない.

演算子オーバーロードによって区間演算を行えるC++のライブラリがある.

kv - a C++ Library for Verified Numerical Computation

作成者のwebサイト.数値計算の専門家.

区間演算の実装について(1) - kashiの日記

この記事が面白かった.

調和級数の部分和で遊んでみた - kashiの日記

非整数に対する数値計算による厳密な結果というものについて驚くほど何も知らないことに気付く.円周率の計算の世界記録が具体的に何をやっているか実際のところを何も知らない.何桁目の数が何であるか確定させるのはどんな誤差評価に基づいているのだろうか.どんな形式で数値を持つのだろうか.

Rust本読めてない! また明日.