Разделить и получить остатки одновременно?
По-видимому, x86 (и, вероятно, множество других наборов инструкций) помещают как частное, так и остальное в операцию разделения в отдельных регистрах.
Теперь мы можем, вероятно, доверять компиляторам, чтобы оптимизировать такой код, чтобы использовать только один вызов для разделения:
( x / 6 )
( x % 6 )
И они, вероятно, это делают. Тем не менее, какие-либо языки (или библиотеки, но в основном ищущие языки) поддерживают одновременное предоставление как результатов разделения, так и по модулю? Если да, то каковы они и каков синтаксис?
Ответы
Ответ 1
C имеет div
и ldiv
. Независимо от того, будут ли они генерировать отдельные инструкции для частного и остального, это будет зависеть от вашей конкретной стандартной реализации библиотеки и настроек компилятора и оптимизации. Начиная с C99, вы также имеете lldiv
для больших чисел.
Ответ 2
Python делает.
>>> divmod(9, 4)
(2, 1)
Это нечетно, потому что Python - это язык высокого уровня.
Итак, Ruby:
11.divmod(3) #=> [3, 2]
* РЕДАКТИРОВАТЬ *
Следует отметить, что цель этих операторов, вероятно, не в том, чтобы делать работу максимально эффективно, скорее всего, существуют функции для правильности/переносимости.
Для тех, кого это интересует, я считаю это код реализации Python для целочисленного divmod:
static enum divmod_result
i_divmod(register long x, register long y,
long *p_xdivy, long *p_xmody)
{
long xdivy, xmody;
if (y == 0) {
PyErr_SetString(PyExc_ZeroDivisionError,
"integer division or modulo by zero");
return DIVMOD_ERROR;
}
/* (-sys.maxint-1)/-1 is the only overflow case. */
if (y == -1 && UNARY_NEG_WOULD_OVERFLOW(x))
return DIVMOD_OVERFLOW;
xdivy = x / y;
/* xdiv*y can overflow on platforms where x/y gives floor(x/y)
* for x and y with differing signs. (This is unusual
* behaviour, and C99 prohibits it, but it allowed by C89;
* for an example of overflow, take x = LONG_MIN, y = 5 or x =
* LONG_MAX, y = -5.) However, x - xdivy*y is always
* representable as a long, since it lies strictly between
* -abs(y) and abs(y). We add casts to avoid intermediate
* overflow.
*/
xmody = (long)(x - (unsigned long)xdivy * y);
/* If the signs of x and y differ, and the remainder is non-0,
* C89 doesn't define whether xdivy is now the floor or the
* ceiling of the infinitely precise quotient. We want the floor,
* and we have it iff the remainder sign matches y's.
*/
if (xmody && ((y ^ xmody) < 0) /* i.e. and signs differ */) {
xmody += y;
--xdivy;
assert(xmody && ((y ^ xmody) >= 0));
}
*p_xdivy = xdivy;
*p_xmody = xmody;
return DIVMOD_OK;
}
Ответ 3
В С#/.NET у вас есть Math.DivRem
:
http://msdn.microsoft.com/en-us/library/system.math.divrem.aspx
Но согласно этой теме это не такая оптимизация.
Ответ 4
Общий Lisp делает: http://www.lispworks.com/documentation/HyperSpec/Body/f_floorc.htm
Ответ 5
Как упоминал Stringer Bell, существует DivRem
, который не оптимизирован до .NET 3.5.
В .NET 4.0 он использует NGen.
Результаты, полученные мной с помощью Math.DivRem
(debug; release = ~ 11000ms)
11863
11820
11881
11859
11854
Результаты, полученные с помощью MyDivRem
(debug; release = ~ 11000ms)
29177
29214
29472
29277
29196
Проект предназначен для x86.
Math.DivRem
Пример использования
int mod1;
int div1 = Math.DivRem(4, 2, out mod1);
Подписи метода
DivRem(Int32, Int32, Int32&) : Int32
DivRem(Int64, Int64, Int64&) : Int64
Код .NET 4.0
[TargetedPatchingOptOut("Performance critical to inline across NGen image boundaries")]
public static int DivRem(int a, int b, out int result)
{
result = a % b;
return (a / b);
}
.NET 4.0 IL
.custom instance void System.Runtime.TargetedPatchingOptOutAttribute::.ctor(string) = { string('Performance critical to inline across NGen image boundaries') }
.maxstack 8
L_0000: ldarg.2
L_0001: ldarg.0
L_0002: ldarg.1
L_0003: rem
L_0004: stind.i4
L_0005: ldarg.0
L_0006: ldarg.1
L_0007: div
L_0008: ret
Ссылка MSDN
Ответ 6
Структура .NET имеет Math.DivRem
:
int mod, div = Math.DivRem(11, 3, out mod);
// mod = 2, div = 3
Хотя DivRem
- это просто оболочка вокруг примерно такого:
int div = x / y;
int mod = x % y;
(я не знаю, может ли дрожание/может оптимизировать эту вещь в одной команде.)
Ответ 7
FWIW, Haskell имеет divMod
и quotRem
, последний из которых соответствует непосредственно машинной команде (в соответствии с интегральными операторами quot vs. div), а divMod
может не быть.
Ответ 8
int result,rest;
_asm
{
xor edx, edx // pone edx a cero; edx = 0
mov eax, result// eax = 2AF0
mov ecx, radix // ecx = 4
div ecx
mov val, eax
mov rest, edx
}
Ответ 9
Это возвращает результат и остаток
int result,rest;
_asm
{
xor edx, edx // pone edx a cero; edx = 0
mov eax, result// eax = 2AF0
mov ecx, radix // ecx = 4
div ecx
mov val, eax
mov rest, edx
}
Ответ 10
В Java класс BigDecimal
имеет операцию divideAndRemainder
, возвращающую массив из 2 элементов с результатом и остатком деления.
BigDecimal bDecimal = ...
BigDecimal[] result = bDecimal.divideAndRemainder(new BigDecimal(60));
Javadoc: https://docs.oracle.com/javase/7/docs/api/java/math/BigDecimal.html#divideAndRemainder(java.math.BigDecimal)