В вычислении, то операция по модулю возвращает остаток или подписанный остаток от деления, после того, как одно число делится на другое ( так называемый модуль операции).
Принимая во внимание два положительного числа и п, по модулю п (сокращенно в моды п) является остаток от евклидового деления из по п, где является дивидендом и п является делителем. Операцию по модулю следует отличать от символа mod, который относится к модулю (или делителю), с которым работает.
Например, выражение «5 mod 2» будет оцениваться как 1, потому что 5, разделенное на 2, дает частное 2, а остаток - 1, а «9 mod 3» будет оцениваться как 0, потому что деление 9 на 3 дает частное 3 и остаток 0; после умножения 3 на 3 из 9 нечего вычитать.
Несмотря на то, как правило, выполнены с и п оба являются целыми числами, многие вычислительные системы позволяют теперь другие типы числовых операндов. Диапазон значений для операции по модулю целого числа от п равно от 0 до п - 1 включительно ( моды 1 всегда 0; моды 0 не определены, возможно, в результате чего деления на ноле ошибок в некоторых языках программирования ). См. « Модульная арифметика» для более старого и связанного с ним соглашения, применяемого в теории чисел.
Когда ровно одно из a или n отрицательно, наивное определение не работает, и языки программирования различаются по способу определения этих значений.
В математике результатом операции по модулю является класс эквивалентности, и любой член этого класса может быть выбран в качестве представителя ; однако обычным представителем является наименьший положительный остаток, наименьшее неотрицательное целое число, которое принадлежит этому классу (то есть остаток от евклидова деления ). Однако возможны и другие соглашения. Компьютеры и калькуляторы имеют различные способы хранения и представления чисел; таким образом, их определение операции по модулю зависит от языка программирования или базового оборудования.
Почти во всех вычислительных системах частное q и остаток r от деления a на n удовлетворяют следующим условиям:
| ( 1) |
Однако это по-прежнему оставляет неоднозначность знака, если остаток не равен нулю: происходят два возможных выбора для остатка, один отрицательный, а другой положительный, и два возможных выбора для частного. В теории чисел всегда выбирается положительный остаток, но в вычислениях языки программирования выбирают в зависимости от языка и знаков a или n. Стандартный Паскаль и АЛГОЛ 68, например, дают положительный остаток (или 0) даже для отрицательных делителей, а некоторые языки программирования, такие как C90, оставляют это на усмотрение реализации, когда любое из n или a отрицательно (см. Таблицу под § Подробности на языках программирования). по модулю 0 не определен в большинстве систем, хотя некоторые из них действительно определяют его как.
или эквивалентно
где sgn - знаковая функция, и, следовательно,
По словам Лейена,
Буте утверждает, что евклидово деление превосходит другие с точки зрения регулярности и полезных математических свойств, хотя разделение по полу, предложенное Кнутом, также является хорошим определением. Несмотря на широкое распространение, усеченное деление уступает другим определениям.
- Даан Лейен, Отделение и модуль для компьютерных ученыхОднако усеченное деление удовлетворяет тождеству.
Некоторые калькуляторы имеют функциональную кнопку mod (), и многие языки программирования имеют аналогичную функцию, например, выраженную как mod ( a, n). Некоторые также поддерживают выражения, которые используют «%», «mod» или «Mod» в качестве оператора по модулю или остатку, например a % n
или a mod n
.
Для сред, в которых отсутствует аналогичная функция, можно использовать любое из трех приведенных выше определений.
Когда результат операции по модулю имеет знак делимого (определение усечения), это может привести к неожиданным ошибкам.
Например, чтобы проверить, является ли целое число нечетным, можно было бы проверить, равен ли остаток на 2 1:
bool is_odd(int n) { return n % 2 == 1; }
Но на языке, где modulo имеет знак делимого, это неверно, потому что, когда n (делимое) является отрицательным и нечетным, n mod 2 возвращает -1, а функция возвращает false.
Одна из правильных альтернатив - проверить, что остаток не равен 0 (потому что остаток 0 одинаков независимо от знаков):
bool is_odd(int n) { return n % 2 != 0; }
Другой альтернативой является использование того факта, что для любого нечетного числа остаток может быть либо 1, либо -1:
bool is_odd(int n) { return n % 2 == 1 || n % 2 == -1; }
Операции по модулю могут быть реализованы так, что деление с остатком вычисляется каждый раз. Для особых случаев на некотором оборудовании существуют более быстрые альтернативы. Например, модуль степеней двойки может быть альтернативно выражен как побитовая операция И (при условии, что x является положительным целым числом, или с использованием определения без усечения):
x % 2n == x amp; (2n - 1)
Примеры:
x % 2 == x amp; 1
x % 4 == x amp; 3
x % 8 == x amp; 7
В устройствах и программном обеспечении, которые реализуют побитовые операции более эффективно, чем по модулю, эти альтернативные формы могут привести к более быстрым вычислениям.
Оптимизация компилятора может распознавать выражения формы, expression % constant
где constant
- степень двойки, и автоматически реализовывать их как expression amp; (constant-1)
, позволяя программисту писать более четкий код без ущерба для производительности. Эта простая оптимизация невозможна для языков, в которых результат операции по модулю имеет знак делимого (включая C ), за исключением случаев, когда делимое имеет целочисленный тип без знака. Это потому, что, если дивиденд отрицательный, модуль будет отрицательным, тогда как expression amp; (constant-1)
всегда будет положительным. Для этих языков вместо этого следует использовать эквивалентность, выраженную с помощью побитовых операций ИЛИ, НЕ и И. x % 2n == x lt; 0 ? x | ~(2n - 1) : x amp; (2n - 1)
Оптимизация для общих операций с постоянным модулем также существует путем вычисления деления сначала с использованием оптимизации с постоянным делителем.
Некоторые операции по модулю можно разложить или разложить на множители аналогично другим математическим операциям. Это может быть полезно в криптографических доказательствах, таких как обмен ключами Диффи – Хеллмана.
Язык | Оператор | Целое число | Плавающая запятая | Определение |
---|---|---|---|---|
ABAP | MOD | да | да | Евклидово |
ActionScript | % | да | Нет | Усеченный |
Ада | mod | да | Нет | Пол |
rem | да | Нет | Усеченный | |
АЛГОЛ 68 | ÷× , mod | да | Нет | Евклидово |
AMPL | mod | да | Нет | Усеченный |
APL | | | да | Нет | Пол |
AppleScript | mod | да | Нет | Усеченный |
AutoLISP | (rem d n) | да | Нет | Усеченный |
AWK | % | да | Нет | Усеченный |
БАЗОВЫЙ | Mod | да | Нет | Неопределенный |
до н.э | % | да | Нет | Усеченный |
C C ++ | % , div | да | Нет | Усеченный |
fmod (C) std::fmod (C ++) | Нет | да | Усеченный | |
remainder (C) std::remainder (C ++) | Нет | да | Закругленный | |
C # | % | да | да | Усеченный |
Clarion | % | да | Нет | Усеченный |
Чистый | rem | да | Нет | Усеченный |
Clojure | mod | да | Нет | Пол |
rem | да | Нет | Усеченный | |
КОБОЛ | FUNCTION MOD | да | Нет | Пол |
CoffeeScript | % | да | Нет | Усеченный |
%% | да | Нет | Пол | |
Холодный синтез | % , MOD | да | Нет | Усеченный |
Common Lisp | mod | да | да | Пол |
rem | да | да | Усеченный | |
Кристалл | % | да | Нет | Усеченный |
D | % | да | да | Усеченный |
Дротик | % | да | да | Евклидово |
remainder() | да | да | Усеченный | |
Эйфель | \\ | да | Нет | Усеченный |
Эликсир | rem/2 | да | Нет | Усеченный |
Integer.mod/2 | да | Нет | Пол | |
Вяз | modBy | да | Нет | Пол |
remainderBy | да | Нет | Усеченный | |
Erlang | rem | да | Нет | Усеченный |
math:fmod/2 | Нет | да | Усечено (то же, что и C) | |
Эйфория | mod | да | Нет | Пол |
remainder | да | Нет | Усеченный | |
F # | % | да | да | Усеченный |
Фактор | mod | да | Нет | Усеченный |
FileMaker | Mod | да | Нет | Пол |
Четвертый | mod | да | Нет | Реализация определена |
fm/mod | да | Нет | Пол | |
sm/rem | да | Нет | Усеченный | |
Фортран | mod | да | да | Усеченный |
modulo | да | да | Пол | |
Фринк | mod | да | Нет | Пол |
GLSL | % | да | Нет | Неопределенный |
mod | Нет | да | Пол | |
GameMaker Studio (GML) | mod , % | да | Нет | Усеченный |
GDScript (Годо) | % | да | Нет | Усеченный |
fmod | Нет | да | Усеченный | |
posmod | да | Нет | Пол | |
fposmod | Нет | да | Пол | |
Идти | % | да | Нет | Усеченный |
math.Mod | Нет | да | Усеченный | |
Groovy | % | да | Нет | Усеченный |
Haskell | mod | да | Нет | Пол |
rem | да | Нет | Усеченный | |
Data.Fixed.mod' ( GHC ) | Нет | да | Пол | |
Haxe | % | да | Нет | Усеченный |
HLSL | % | да | да | Неопределенный |
J | | | да | Нет | Пол |
Джава | % | да | да | Усеченный |
Math.floorMod | да | Нет | Пол | |
JavaScript TypeScript | % | да | да | Усеченный |
Юлия | mod | да | Нет | Пол |
% , rem | да | Нет | Усеченный | |
Котлин | % , rem | да | да | Усеченный |
mod | да | да | Пол | |
кш | % | да | Нет | Усечено (то же, что и POSIX sh) |
fmod | Нет | да | Усеченный | |
LabVIEW | mod | да | да | Усеченный |
LibreOffice | =MOD() | да | Нет | Пол |
Логотип | MODULO | да | Нет | Пол |
REMAINDER | да | Нет | Усеченный | |
Lua 5 | % | да | да | Пол |
Lua 4 | mod(x,y) | да | да | Усеченный |
Liberty BASIC | MOD | да | Нет | Усеченный |
Mathcad | mod(x,y) | да | Нет | Пол |
Клен | e mod m (по умолчанию), modp(e, m) | да | Нет | Евклидово |
mods(e, m) | да | Нет | Закругленный | |
frem(e, m) | да | да | Закругленный | |
Mathematica | Mod[a, b] | да | Нет | Пол |
MATLAB | mod | да | Нет | Пол |
rem | да | Нет | Усеченный | |
Максима | mod | да | Нет | Пол |
remainder | да | Нет | Усеченный | |
Встроенный язык Maya | % | да | Нет | Усеченный |
Майкрософт Эксель | =MOD() | да | да | Пол |
Minitab | MOD | да | Нет | Пол |
Модула-2 | MOD | да | Нет | Пол |
REM | да | Нет | Усеченный | |
Швабры | # | да | Нет | Пол |
Сетевой ассемблер ( NASM, NASMX ) | % , div (без знака) | да | Нет | N / A |
%% (подписано) | да | Нет | Определяется реализацией | |
Ним | mod | да | Нет | Усеченный |
Оберон | MOD | да | Нет | Напольный |
Цель-C | % | да | Нет | Усеченный (такой же, как C99) |
Object Pascal, Delphi | mod | да | Нет | Усеченный |
OCaml | mod | да | Нет | Усеченный |
mod_float | Нет | да | Усеченный | |
Оккам | \ | да | Нет | Усеченный |
Паскаль (ISO-7185 и -10206) | mod | да | Нет | Евклидово |
Расширенный код программирования ( PCA ) | \ | да | Нет | Неопределенный |
Perl | % | да | Нет | Пол |
POSIX::fmod | Нет | да | Усеченный | |
Фикс | mod | да | Нет | Пол |
remainder | да | Нет | Усеченный | |
PHP | % | да | Нет | Усеченный |
fmod | Нет | да | Усеченный | |
PIC BASIC Pro | \\ | да | Нет | Усеченный |
PL / I | mod | да | Нет | Напольный (ANSI PL / I) |
PowerShell | % | да | Нет | Усеченный |
Код программирования ( PRC ) | MATH.OP - 'MOD; (\)' | да | Нет | Неопределенный |
Прогресс | modulo | да | Нет | Усеченный |
Пролог ( ISO 1995 ) | mod | да | Нет | Пол |
rem | да | Нет | Усеченный | |
PureBasic | % , Mod(x,y) | да | Нет | Усеченный |
PureScript | `mod` | да | Нет | Пол |
Чистые данные | % | да | Нет | Усечено (то же, что и C) |
mod | да | Нет | Пол | |
Python | % | да | да | Пол |
math.fmod | Нет | да | Усеченный | |
Q # | % | да | Нет | Усеченный |
р | %% | да | Нет | Пол |
Ракетка | modulo | да | Нет | Пол |
remainder | да | Нет | Усеченный | |
Раку | % | Нет | да | Пол |
RealBasic | MOD | да | Нет | Усеченный |
Причина | mod | да | Нет | Усеченный |
Rexx | // | да | да | Усеченный |
РПГ | %REM | да | Нет | Усеченный |
Рубин | % , modulo() | да | да | Пол |
remainder() | да | да | Усеченный | |
Ржавчина | % | да | да | Усеченный |
rem_euclid() | да | да | Евклидово | |
SAS | MOD | да | Нет | Усеченный |
Скала | % | да | Нет | Усеченный |
Схема | modulo | да | Нет | Пол |
remainder | да | Нет | Усеченный | |
Схема R 6 RS | mod | да | Нет | Евклидово |
mod0 | да | Нет | Закругленный | |
flmod | Нет | да | Евклидово | |
flmod0 | Нет | да | Закругленный | |
Царапать | mod | да | Нет | Пол |
mod | Нет | да | Усеченный | |
Семя7 | mod | да | да | Пол |
rem | да | да | Усеченный | |
SenseTalk | modulo | да | Нет | Пол |
rem | да | Нет | Усеченный | |
sh (POSIX) (включает bash, mksh и т. Д.) | % | да | Нет | Усечено (то же, что и C) |
Болтовня | \\ | да | Нет | Пол |
rem: | да | Нет | Усеченный | |
Щелчок! | mod | да | Нет | Пол |
Вращаться | // | да | Нет | Пол |
Твердость | % | да | Нет | Пол |
SQL ( SQL: 1999 ) | mod(x,y) | да | Нет | Усеченный |
SQL ( SQL: 2011 ) | % | да | Нет | Усеченный |
Стандартный ML | mod | да | Нет | Пол |
Int.rem | да | Нет | Усеченный | |
Real.rem | Нет | да | Усеченный | |
Stata | mod(x,y) | да | Нет | Евклидово |
Быстрый | % | да | Нет | Усеченный |
truncatingRemainder(dividingBy:) | Нет | да | Усеченный | |
Tcl | % | да | Нет | Пол |
Крутящий момент | % | да | Нет | Усеченный |
Тьюринг | mod | да | Нет | Пол |
Verilog (2001) | % | да | Нет | Усеченный |
VHDL | mod | да | Нет | Пол |
rem | да | Нет | Усеченный | |
VimL | % | да | Нет | Усеченный |
Visual Basic | Mod | да | Нет | Усеченный |
WebAssembly | i32.rem_s , i64.rem_s | да | Нет | Усеченный |
сборка x86 | IDIV | да | Нет | Усеченный |
XBase ++ | % | да | да | Усеченный |
Mod() | да | да | Пол | |
Инструмент доказательства теорем Z3 | div , mod | да | Нет | Евклидово |
Кроме того, многие компьютерные системы предоставляют divmod
функцию, которая позволяет одновременно вычислять частное и остаток. Примеры включают в архитектуре x86 «сек IDIV
инструкции, язык программирования C в div()
функции и Python » s divmod()
функцию.
Иногда бывает полезно для результата в по модулю п лежать не между 0 и п - 1, но между некоторым числом д и д + п - 1. В этом случае d называется смещением. Там, кажется, не являются стандартным обозначением для этой операции, так что давайте предварительно использовать в модах д н. Таким образом, мы имеем следующее определение: x = a mod d n на всякий случай d ≤ x ≤ d + n - 1 и x mod n = a mod n. Ясно, что обычная операция по модулю соответствует смещению нуля: a mod n = a mod 0 n. Операция по модулю со смещением связана с функцией пола следующим образом:
(Чтобы убедиться в этом, пусть. Сначала мы покажем, что x mod n = a mod n. В общем случае верно, что ( a + bn) mod n = a mod n для всех целых чисел b ; таким образом, это верно также в частном случай, когда ; но это означает, что это то, что мы хотели доказать. Осталось показать, что d ≤ x ≤ d + n - 1. Пусть k и r - такие целые числа, что a - d = kn + r с 0 ≤ r ≤ n - 1 (см. Евклидово деление ). Тогда, таким образом. Теперь возьмем 0 ≤ r ≤ n - 1 и прибавим d к обеим сторонам, получив d ≤ d + r ≤ d + n - 1. Но мы видели что x = d + r, так что все готово. □)
Модуль со смещением a mod d n реализован в системе Mathematica как Mod[a, n, d]
.
Несмотря на математическую элегантность напольного деления Кнута и евклидова деления, в языках программирования, как правило, гораздо чаще можно найти усеченное деление по модулю на основе деления. Лейен предлагает следующие алгоритмы для вычисления двух делений с учетом усеченного целочисленного деления:
/* Euclidean and Floored divmod, in the style of C's ldiv() */ typedef struct { /* This structure is part of the C stdlib.h, but is reproduced here for clarity */ long int quot; long int rem; } ldiv_t; /* Euclidean division */ inline ldiv_t ldivE(long numer, long denom) { /* The C99 and C++11 languages define both of these as truncating. */ long q = numer / denom; long r = numer % denom; if (r lt; 0) { if (denom gt; 0) { q = q - 1; r = r + denom; } else { q = q + 1; r = r - denom; } } return (ldiv_t){.quot = q,.rem = r}; } /* Floored division */ inline ldiv_t ldivF(long numer, long denom) { long q = numer / denom; long r = numer % denom; if ((r gt; 0 amp;amp; denom lt; 0) || (r lt; 0 amp;amp; denom gt; 0)) { q = q - 1; r = r + denom; } return (ldiv_t){.quot = q,.rem = r}; }
Обратите внимание, что в обоих случаях остаток можно вычислить независимо от частного, но не наоборот. Здесь операции объединены для экономии места на экране, так как логические ветви одинаковы.