剰余演算
[Wikipedia|▼Menu]
アルゴリズムの違いによる商(赤線)と余り(緑線)のグラフ

剰余演算(じょうよえんざん、リマインダーまたはモジュロとも呼ぶ)は、ある数値を別の数値で除算し、余りを取得する演算である[1][2]
概要

2つの正の整数である、被除数 a および 除数 n が与えられた場合、a の n による剰余(a modulo n、略して a mod n とも表記される)は、ユークリッド除法における a を n で除算した余りである。例えば、「5 mod 2」の結果は 1 である。5を2で除算した場合、は2となり、余りは1となるからである。また、「9 mod 3」の結果は 0 である。9を3で除算した商は3となり、余りは0となる(つまり9から3を3回引くと残りがなくなる)からである[3]

通常の場合、a と n はともに整数であるが、多くのコンピュータシステムでは他の数値型でも処理が可能である。整数 n の剰余の取りうる範囲は、0 から n − 1 までである。「n mod 1」 の場合は常に 0 となる。「n mod 0」は未定義であり、プログラミング言語によっては「ゼロ除算」エラーを引き起こす。a または n が負数の場合については、単純な定義はなく、プログラミング言語によってどのように定義されるかが異なっている。

数論における古典的な関連事項については合同算術を参照。
剰余演算による余りの算出

数学的には、剰余演算の結果はユークリッド除法における余りのことである。しかし、別の法則に従って算出されることもある。コンピュータやその他の計算機は数値の保持や処理方法がさまざまなので、剰余演算の定義はプログラミング言語や動作するハードウェアによって、それぞれ規定されている。

ほぼすべてのコンピュータシステムにおいて、a を n で除算した商 q および剰余 r は下記の条件を満たす。 q ∈ Z a = n q + r 。 r 。 < 。 n 。 {\displaystyle {\begin{aligned}q\,&\in \mathbb {Z} \\a\,&=nq+r\\|r|\,&<|n|\end{aligned}}} (1)

ところが、剰余が0でない場合、その符号は不確定なものとなる。数論上は、余りは常に正の数となるが、プログラミング言語によっては a および n の符号によって剰余の符号が正となるか負となるかが定められる。標準的なPascalおよびAlgol68では、除数が負数であっても正の剰余(または0)を出力する。しかし、C90のようなプログラミング言語では、n または a が負の数の場合にはそうならないことがある。詳細は右表を参照。多くのシステムでは a の 0 での剰余は未定義だが、いくつかは a を出力するように定義されている。

多くの実装においては「0への切捨て除算」が使用されている。これは商を0への切捨て関数 q = trunc(a/n) によって処理し、(1)の処理における余りは「被除数と同じ符号」となる。この場合の商は0方向に丸められる。つまり、実数での商から0の方向へ向かって直近の整数となる。

ドナルド・クヌースは「切り下げ除算」について言及した[4]。これは商を床関数 q = ⌊ ( a / n ) ⌋ {\displaystyle q=\lfloor (a/n)\rfloor } によって処理し、(1)の処理における余りは「除数と同じ符号」となる。床関数によって、商が負の数であっても常に負の無限大方向に丸められる。 r = a − n ⌊ a n ⌋ {\displaystyle r=a-n\left\lfloor {\frac {a}{n}}\right\rfloor }

Raymond T. Bouteはユークリッド除法の手法での定義について言及している[5]。この場合、余りは非負数(0 ? r)となる。右表では「常に正」と表記している。この定義は下記のように表すことができる。 n > 0 ⇒ q = ⌊ a n ⌋ {\displaystyle n>0\Rightarrow q=\left\lfloor {\frac {a}{n}}\right\rfloor } n < 0 ⇒ q = ⌈ a n ⌉ {\displaystyle n<0\Rightarrow q=\left\lceil {\frac {a}{n}}\right\rceil }

また、符号関数 sgn を使用すると、下記のようにも表せる。 q = sgn ⁡ ( n ) ⌊ a 。 n 。 ⌋ {\displaystyle q=\operatorname {sgn}(n)\left\lfloor {\frac {a}{\left|n\right|}}\right\rfloor } r = a − 。 n 。 ⌊ a 。 n 。 ⌋ {\displaystyle r=a-|n|\left\lfloor {\frac {a}{\left|n\right|}}\right\rfloor }

Common Lispでは切り下げ除算、切り上げ除算のそれぞれについてq = round(a/n) と q = ceil(a/n) で商が与えられる。

IEEE 754-1985 の剰余は、近接丸め(英語版)を行った場合の商として定義している。したがって、剰余の符号は「0に近い側」となる。

Daan Leijenはこのように記している。.mw-parser-output .templatequote{overflow:hidden;margin:1em 0;padding:0 40px}.mw-parser-output .templatequote .templatequotecite{line-height:1.5em;text-align:left;padding-left:1.6em;margin-top:0}Bouteはユークリッド除法が数学的に一般的で有用なので他の方法よりも優れていることを示した。しかし、クヌースの示した切り下げ除算もまたよい定義である。「0への切捨て除算」は幅広く使われているが、他の定義よりも劣っている。—Daan Leijen、Division and Modulus for Computer Scientists[6]
剰余演算のバリエーション// 商を求める関数function truncated_division(divisor, dividend) { return Math.trunc(divisor / dividend);}function floored_division(divisor, dividend) { return Math.floor(divisor / dividend);}function euclidean_division(divisor, dividend) { return Math.sign(dividend) * Math.floor(divisor / Math.abs(dividend));}function ceiled_division(divisor, dividend) { return Math.ceil(divisor / dividend);}function rounded_division(divisor, dividend) { return Math.round(divisor / dividend);}// 剰余を求める関数function truncated_mod(divisor, dividend) { return divisor - dividend * truncated_division(divisor, dividend);}function floored_mod(divisor, dividend) { return divisor - dividend * floored_division(divisor, dividend);}function euclidean_mod(divisor, dividend) { return divisor - dividend * euclidean_division(divisor, dividend);}function ceiled_mod(divisor, dividend) { return divisor - dividend * ceiled_division(divisor, dividend);}function rounded_mod(divisor, dividend) { return divisor - dividend * rounded_division(divisor, dividend);}


次ページ
記事の検索
おまかせリスト
▼オプションを表示
ブックマーク登録
mixiチェック!
Twitterに投稿
オプション/リンク一覧
話題のニュース
列車運行情報
暇つぶしWikipedia

Size:36 KB
出典: フリー百科事典『ウィキペディア(Wikipedia)
担当:undef