Общие советы по представлению больших чисел

21

Иногда во время игры в гольф нужно представлять большое количество кода в своем коде. Запись их как есть может значительно увеличить количество байтов.

Какие общие 1 советы вы имеете для краткого представления длинных чисел в коде?

Пожалуйста, оставьте один совет за ответ.


1 Под общими я подразумеваю советы, которые могут быть применены к более чем одному языку. Советы по конкретным языкам публикуются в соответствующей ветке.

Арджун
источник
Связанный.
Мартин Эндер
Я видел, что кто-то неправильно понял вопрос - возможно, в названии должно быть сказано, что речь идет о гольфе.
Орджан Йохансен

Ответы:

15

Ищите специальные номера

Некоторые языки имеют встроенные функции для квадратов, возведения в степень с основанием 2, n-го простого, факториального или других процедур, которые могут генерировать большие числа. Проверьте, попадает ли ваш номер в какую-либо из этих категорий.

И если это не так, может случиться так, что большее число, которое подходит для ваших целей и может быть использовано вместо.

Луис Мендо
источник
4
Даже если желаемое число не может быть сгенерировано с помощью встроенной функции, возможно, можно сгенерировать число, которое можно использовать в качестве основы для простого вычисления, чтобы получить ваш номер, при этом сохраняя байты при его записи.
Лохматый
7
+1 для «большего числа» - если 1.01e6итераций достаточно, 1e7экономит 3 байта за счет времени выполнения.
Крис Х
13

Используйте побитовые логические операторы

У некоторых языков есть побитовое И, ИЛИ, XOR, а иногда НЕТ.

Выражение определенного большого числа в виде побитовой комбинации результата возведения в степень или сдвига влево и другого числа может привести вас к именно тому числу, которое вам нужно. Обычно это того стоит, если цифры становятся достаточно большими.

Например, 2147483722это 10 байтов, но 2<<30^74(2 ^ 31 бит-XORed с 74) только 8.

L3viathan
источник
3
Оператор сдвига может быть заменен возведением в степень, если в языке есть этот оператор (например bc). И XOR никогда не бывает более полезным, чем +и -: в этом случае xor и add дают одинаковый результат, но во всех случаях есть некоторое целое число, которое можно сложить или вычесть, чтобы получить тот же результат, что и xor с целым числом, и добавление не больше, а иногда и короче.
Ричи
3
@rici Истинно, если все целые числа записаны как простые десятичные дроби, но вполне возможно, что целое число, которое будет использоваться с xor, может быть сокращено таким образом, что целое число, которое будет использоваться с плюсом или минусом, не может: думать о чем-то вроде 1e9^2e9.
HVD
2
@rici Как бы вы выразились, 9<<49^7<<19используя сложение вместо xor?
L3viathan
2
@rici Нет, я имел в виду то, что написал. Он оценивает 1286561280в JavaScript и Perl (и, возможно, в других языках), и это более короткое выражение для получения этого значения, чем эквивалентное использование +или -.
HVd
@hvd: Хорошо, точка взята.
Ричи
11

Используйте строки для повторяющихся чисел

Для чисел, которые очень повторяются по своей природе, вы можете использовать строки и привести их к целому числу. Например, в JavaScript

+"1".repeat(100) // returns 100 1s (saves 84 bytes!)
Арджун
источник
2
Или вы можете использовать очень большую дробь, 1e100/9в этом случае.
ETHproductions
11

Используйте научную нотацию

Научная запись может сохранять байты в случае длинных чисел. Например:

3564e-8 // returns 0.00003564 (saves 3 bytes!)
Арджун
источник
3
Почему бы просто не использовать 3564e-8в этом случае?
Джои
@ Джо Да, конечно! Спасибо! :)
Арджун
Конечно, вы обычно можете написать другое число как .00003564, которое также на один байт короче снова.
Джои
@Joey Не каждый язык делает, следовательно, я не буду редактировать это в.
Арджун
Это преднамеренно, что 3564 (слева) становится 354 (справа), или это опечатка?
Эрик
10

Ищите другой номер, чтобы использовать вместо

Это может звучать как отсутствие ответа, но не всегда очевидно, что большее число может быть вычислено с помощью более короткого кода. Пример, который я помню, это Вывод googol-копий строки , где очевидные ответы требуют вычисления 10 100 . Как выясняется, вычисление любого кратного 10 100 приводит к одинаково правильному, но в некоторых языках более короткому ответу. Ответ Денниса там использует 100 100 , мой собственный использует 250 255 .

HVD
источник
4
Другим примером этого является CJam, где вы можете получить текущую временную метку, esесли вам просто нужно большое число, но вам не важно его значение (или оно всегда одинаковое)
Мартин Эндер
10

Базовое сжатие

Базовый декомпрессионный код может быть довольно сложным, но если у вас действительно огромное количество, иногда это может помочь сжать его в некоторой базе выше 10.

Также помогает то, что в некоторых языках базовый код сжатия очень прост. Например, PHP имеет base64_decode(_), Python имеет int(_,36), JavaScript имеет parseInt(_,36), и многие языки гольфа имеют встроенные декомпрессионные встроенные функции. Например, в CJam:

"+ÜTbô±"256b

Это содержит непечатный. Попробуйте онлайн!

Это дает:

12345678987654321
Esolanging Fruit
источник
9

Используйте экспоненциальные дроби для больших повторяющихся чисел

Допустим, вы хотите сгенерировать число, состоящее из 100 единиц. Вы могли бы использовать int("1"*100), +"1".repeat(100)и т.д. , но вы также можете воспользоваться тем , что это очень близко к

1e100/9

Это лучше всего подходит для очень повторяющихся чисел, например, из одной цифры. Пара повторяющихся цифр также работает довольно хорошо:

12e100/99  // Generates 121212121212... (100 digits)

Изредка вы можете найти какой-то другой странный шаблон, который также может быть представлен довольно кратко в этом методе. Если вам это понадобилось int("123456790"*11), например:

1e100/81

Однако будьте осторожны: такие числа int("1234567890"*10)не имеют такого простого представления.

ETHproductions
источник
9

Используйте побитовый сдвиг влево для возведения в степень 2

Хотя существует много языков, которые поддерживают оператор для возведения в степень, некоторые этого не делают. А те, которые этого не делают, обычно требуют вызова функций (или методов класса / объекта), которые могут стоить несколько байтов.

Но вы можете сохранить несколько байтов, когда вам нужно поднять 2 до степени n , используя оператор Bitwise Left Shift <<as 1<<n. Обратите внимание, что это сохранит ваши байты только в том случае, если n больше или равно 17. Однако это всегда будет сохранять ваши байты, если n является динамическим. Несколько примеров:

1<<2 // returns 4 (3 bytes more :( )
1<<3 // returns 8 (3 bytes more :( )
1<<6 // returns 64 (2 bytes more :( )
1<<14 // returns 16384 (no bytes saved)
1<<17 // returns 131072 (saves 1 byte!)
1<<18 // returns 262114 (saves 1 byte!)
Арджун
источник
4
Обратите внимание, что вы можете использовать другие значения вместо 1., 8<<9 // 4096чтобы мы могли получить до 99<<616 байтов, что равняется 6,917,529,027,641,081,856экономии 13 байтов!
Draco18s
@ Draco18s Да, и она покрыта здесь .
Арджун,
Этот еще один охватывает логические побитовые операторы. Да, он также имеет сдвиг битов, но речь идет об использовании двух больших чисел для XOR и получения третьего, конкретного числа.
Draco18s
7

Китайская теорема об остатках

Если часто появляются произвольные большие целые числа или если большое целочисленное представление в целевом языке программирования стоит слишком много байтов, вы можете использовать китайскую теорему об остатках.

Выберите несколько попарно относительно простых чисел m i > = 2, и вы можете выразить большое число от 0 до lcm (m 1 , m 2 , ..., m i ) -1

Например, я выбираю 2, 3, 5, 11, 79, 83, 89, 97, тогда я могу однозначно выразить число меньше 18680171730. 10000000000 (1e10) можно выразить как 0,1,0,1,38,59,50,49 (1e10 mod 2, 3 ..., 97), которые не нужно выражать как специальный класс / структуру Big Integer, которые могут сохранять несколько байтов в некотором языке программирования.

Сложение и вычитание могут быть сделаны непосредственно с использованием этого представления. Пример:

(0,1,0,1,38,59,50,49)+(0,2,0,6,23,20,16,53) = 1e10 + 5000 
                                            = (0+0 mod 2, 1+2 mod 3, 0+0 mod 5, 1+6 mod 11, 38+23 mod 79, 59+20 mod 83, 50+16 mod 89, 49+53 mod 97)
Топор арии
источник
1
Хотите уточнить это?
Скидсдев
@Mayube Я добавил некоторые объяснения, но я сомневаюсь, что в большинстве случаев они бесполезны.
Топор Арии
1
Это «теорема об остатке».
Орджан Йохансен
6

Используйте String Padding (где это возможно)

Если большое число включает в себя повторяющуюся цифру в начале или конце, вы можете сохранить байты, используя один из методов заполнения вашего языка, чтобы создать строку нужного числа, которую затем можно преобразовать в целое число.


пример

Чтобы сгенерировать число 1111111111111111111111112(25 байт) в JavaScript (ES8):

+"2".padStart(25,1) // 19 bytes
мохнатый
источник
3

Используйте экспоненты

Если на вашем языке есть оператор экспоненты, вы можете использовать его для генерации, если не нужного вам числа, хотя бы числа, которое вы можете выполнить простым вычислением или 2, чтобы получить свой номер. Даже без оператора вы все равно сможете сохранять байты с помощью встроенной функции или метода.

пример

Максимально безопасное число в JavaScript является 9007199254740991, длиной 16 цифр. В ES7 это можно рассчитать с помощью следующих 7 байтов:

2**53-1

Эквивалент в ES6 и более ранних версиях, хотя и такой же длины, что и само целое число в этом случае, демонстрирует, что использование более подробного метода может не обязательно стоить вам байтов.

Math.pow(2,53)-1

Выше, однако, может получиться короче, если, например, у вас уже естьMath указали псевдоним для одного символа в другом месте своего кода.

мохнатый
источник
2

Используйте дроби на месте поплавка

Пример: 1./3вместо0.333333333

RosLuP
источник