[Перевод] Лучшая подарочная книга для начитанных фанатов JavaScript

Здравствуйте, уважаемые хаброжители!

Мы вынашиваем амбициозные планы по изданию вот такой книги:

d9fbf7570a4c44ebbe92ace3e20a5d49.jpg

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

Приятного чтения, и поучаствуйте пожалуйста в опросе!

Я написал книгу If Hemingway Wrote JavaScript. В ней я фантазирую, как 25 знаменитых прозаиков, поэтов и драматургов могли бы решать простые задачи на JavaScript. Это дань уважения моим любимым писателям и признание в любви языку JavaScript — ведь я не знаю, какой еще язык наделял бы программиста такой свободой, позволял раскрыть творческий потенциал, а также был бы настолько своеобразен, чтобы привлечь внимание великих литераторов.

В этом посте — оригинальный материал, которого нет в книге (считайте его «бонусом»). Это первый глубокий технический разбор решений, приписываемых каждому автору. Некоторые решения требуют более подробных объяснений, нежели другие.
Приятного чтения!

Часть 1: Простые числа

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

1. Хорхе Луис Борхес

https://github.com/angus-c/literary.js/tree/master/book/borges/prime.js

// Они повествуют (я знаю) о пинаклях, флеронах и балюстрадах,
// о потаенных интервольтах и чудищах, вечно идущих вверх размашистым шагом 
 
var monstersAscendingAStaircase = function(numberOfSteps) {
  var stairs = []; stepsUntrodden = [];
  var largestGait = Math.sqrt(numberOfSteps);
 
  // Череда тварей карабкается по ступенькам;
  // шаг каждой следующей химеры шире, чем у предыдущей 
  for (var i = 2; i <= largestGait; i++) {
    if (!stairs[i]) {
      for (var j = i * i; j <= numberOfSteps; j += i) {
        stairs[j] = "stomp";
      }
    }
  
// Длиннолапые чудища не попадают по ступенькам, номера которых – простые числа 
  for (var i = 2; i <= numberOfSteps; i++) {
    if(!stairs[i]) {
      stepsUntrodden.push(i);
    }
  }
 
  // Вот и наш ответ
  return stepsUntrodden;
};

Решение Борхеса — вариант алгоритма «Решето Эратосфена», в котором кратные каждого известного простого числа помечаются как составные (непростые) числа. В таком случае, у Борхеса длиннолапые чудовища занимают места делителей. Каждое чудище делает шаг на одну ступеньку шире, чем шедшее за ним: 2, 3, 4, 5… вплоть до квадратного корня из числа, соответствующего наивысшей ступеньке. (почему-то Борхес позволяет карабкаться по лестнице и чудищам с неровным шагом). Те ступеньки, на которые никто не встанет — и есть простые числа.

61af342dbafd4e6ea67be74613555f0d.png

Обратите внимание на строку 12: каждый монстр начинает восхождение с квадрата своего множителя:


for (var j = i * i; j <= numberOfSteps; j += i) {

Дело в том, что составные числа между n и n² уже будут пройдены чудищами с более мелким шагом.

2. Льюис Кэрролл

https://github.com/angus-c/literary.js/tree/master/book/carroll/prime.js

function downTheRabbitHole(growThisBig) {
  var theFullDeck = Array(growThisBig);
  var theHatter = Function('return this/4').call(2*2);
  var theMarchHare = Boolean('The frumious Bandersnatch!');
 
  var theVerdict = 'the white rabbit'.split(/the march hare/).slice(theHatter);
 
  // в море слез…
  eval(theFullDeck.join('if (!theFullDeck[++theHatter]) {\
      theMarchHare = 1;\
      theVerdict.push(theHatter);\
      ' + theFullDeck.join('theFullDeck[++theMarchHare * theHatter]=true;') + '}')
  );
 
  return theVerdict;
}

Как и творчество Кэрролла, его код — смесь загадки и нонсенса. Давайте построчно в нем разберемся, начиная с объявления переменных.
В принципе, строка 2 вполне традиционна (если закрыть глаза на использование конструктора Array). Кэрролл создает пустой массив, длина которого совпадает с сообщенным аргументом. Он называется theFullDeck, поскольку в решении воображается колода карт, причем в итоге рубашкой вниз будут лежать лишь те из них, что соответствуют простым числам.

В строке 3 создается функция (с применением малоиспользуемого конструктора Function), а затем эта функция вызывается при помощи call, при этом 2×2 (т.е. 4) передается как аргумент this. Следовательно, theHatter инициализируется со значением 1.

В строке 4 theMarchHare устанавливается в true . Когда конструктор Boolean вызывается как функция, его аргумент преобразуется в true или false . В таком случае непустая строка «The frumious Bandersnatch!» преобразуется в true . (Кстати, такое присваивание не очень-то здесь нужно, поскольку в строке 10 theMarchHare присваивается новое значение).

Наконец — вероятно, это верх абсурда — в строке 6 Кэрролл присваивает theVerdict пустой массив, и делает это максимально иносказательно:


var theVerdict = 'the white rabbit'.split(/the march hare/).slice(theHatter);

Здесь не так много бросается в глаза. Аргумент для split — это регулярное выражение, не совпадающее с «the white rabbit», поэтому при вызове split получаем массив, в котором содержится лишь «the white rabbit». Последующая операция slice заносит в копию массива все элементы исходного массива, начиная с заданного индекса. Поскольку в нашем одноэлементном массиве нет индекса 1 (таково значение theHatter), никакие члены из него не копируются, и у нас получается пустой массив.

Проще говоря, можно переписать объявления переменных вот так:

function downTheRabbitHole(growThisBig) {
  var theFullDeck = Array(growThisBig);
  var theHatter = 1;
  var theMarchHare = true;
  var theVerdict = [];

А теперь полный угар:

// в море слез…
eval(theFullDeck.join('if (!theFullDeck[++theHatter]) {\
    theMarchHare = 1;\
    theVerdict.push(theHatter);\
    ' + theFullDeck.join('theFullDeck[++theMarchHare * theHatter]=true;') + '}')
);

Прежде, чем перейти к пресловутой функции eval, давайте поговорим о вложенных инструкциях join . Функция join превращает массив в строку, при этом ее аргумент служит клеем между элементами массива. Если вызвать join применительно к пустому массиву, получится строка, состоящая из сплошного клея (повторенная n — 1 раз, где n — длина массива):

Array(4).join('hi'); //'hihihi'

Если вложить друг в друга два join, то вкладывается и соответствующий клей:

Array(4).join('A' + Array(4).join('a')); //'AaaaAaaaAaaa'

Когда же мы включаем в клей переменные, начинаем понимать, что к чему:

var arr = [], count = 0;

Array(4).join('arr.push(' + Array(4).join('count++,') + '-1);');
//"arr.push(count++,count++,count++,-1);arr.push(count++,count++,count++,-1);arr.push(count++,count++,count++,-1)"



Теперь, растолковав JavaScript, как генерировать JavaScript, остается только придумать, как все это запустить. Заходим в гнусную eval

var arr = [], count = 0;
eval(Array(4).join('arr.push(' + Array(4).join('count++,') + '-1);'));
arr; //[0, 1, 2, -1, 3, 4, 5, -1, 6, 7, 8, -1]
…так вот каким образом Кэрролл автоматически генерирует программу для работы с простыми числами. Давайте еще раз взглянем на этот код:
        // в море слез...
eval(theFullDeck.join('if (!theFullDeck[++theHatter]) {\
    theMarchHare = 1;\
    theVerdict.push(theHatter);\
    ' + theFullDeck.join('theFullDeck[++theMarchHare * theHatter]=true;') + '}')
);
Аргумент к eval (после форматирования) принимает вид:
        if (!theFullDeck[++theHatter]) {
  theMarchHare = 1;
  theVerdict.push(theHatter);
  theFullDeck[++theMarchHare * theHatter] = true;
  theFullDeck[++theMarchHare * theHatter] = true;
  theFullDeck[++theMarchHare * theHatter] = true;
}
if (!theFullDeck[++theHatter]) {
  theMarchHare = 1;
  theVerdict.push(theHatter);
  theFullDeck[++theMarchHare * theHatter] = true;
  theFullDeck[++theMarchHare * theHatter] = true;
  theFullDeck[++theMarchHare * theHatter] = true;
}
if (!theFullDeck[++theHatter]) {
  theMarchHare = 1;
  theVerdict.push(theHatter);
  theFullDeck[++theMarchHare * theHatter] = true;
  theFullDeck[++theMarchHare * theHatter] = true;
  theFullDeck[++theMarchHare * theHatter] = true;
}
// etc...

…и т.д. (Сгенерированный таким образом код может получиться очень длинным. Запросив все простые числа вплоть до 100, получим более 10 000 строк кода — можете себе представить, как это скажется на производительности –, но для Страны Чудес, полагаю, сойдет)

Итак, постепенно все проясняется. Оказывается, Кэрролл использовал вариант того самого решета Эратосфена, которое мы видели у Борхеса. theFullDeck — это массив со всеми числами, которые необходимо проверить, theHatter и theMarchHare — вложенные счетчики, умножаемые при каждом инкременте, чтобы сгенерировать все возможные составные числа. Все карты, чьи индексы — составные числа, переворачиваются (поскольку theFullDeck с таким индексом равен true). Открытыми остаются лишь те карты, которым соответствуют простые числа.

d765776ba45f426b8b2935cbee4cf892.png

3. Дуглас Адамс

https://github.com/angus-c/literary.js/tree/master/book/adams/prime.js

// Вот я, мозг размером с планету, и они просят меня написать на JavaScript...
function kevinTheNumberMentioner(_){
  l=[]
  /* почти безвредно --> */ with(l) {
 
    // извините за все это, у моей вавилонской рыбки сегодня мигрень...
    for (ll=!+[]+!![];ll<_+(+!![]);ll++) {
      lll=+!![];
      while(ll%++lll);
      // У меня в правом полушарии все болит от этих чертовых точек с запятой
      (ll==lll)&&push(ll);
    }
    forEach(alert);
  }
 
  // ну так вы же точно делать не будете...
  return [!+[]+!+[]+!+[]+!+[]]+[!+[]+!+[]];
}

Читать код Адамса в целом сложно, так как он многое заимствует из jsfuck — хитроумного, но убийственно лаконичного языка, в котором используется всего 6 символов. Тем не менее, это настоящий JavaScript — если вы запустите его в консоли, все сработает. Давайте переведем простой фрагмент:

for (ll=!+[]+!![];ll<_+(+!![]);ll++) {

Здесь видим цикл for, а ll и _ — имена переменных. Все остальное — литерал и форменный вынос мозга.
В первом условии этой инструкции ll получает значение !+[]+!![] . Разобрав это выражение, видим в нем два пустых литерала массива. Перед первым стоит +, из-за которого массив принудительно приводится к числу 0. Прямо перед ним стоит!, принудительно приводящий 0 к его булевской противоположности, то есть, к true . Итак, !+[] результирует в true .

Теперь рассмотрим второй литерал массива. Ему предшествуют два !!, что попросту принудительно приведет его к булеану. Поскольку массивы — всегда объекты, булеан массива всегда равен true . Итак, !![] всегда дает true .

Сложив два этих выражения, !+[]+!![], фактически, имеем true + true . Здесь + принудительно приводит оба операнда к числу 1, поэтому окончательный результат всего выражения равен 2.
Два остальных условия цикла for теперь понятны без труда. Опять же, имеем !![], на сей раз перед ним идет +, принудительно приводящий true к 1. Итак, ll<_+(+!![]) дает ll < _ + 1 .
Последнее условие — это обычный постфикс JavaScript, поэтому весь цикл for дает:

for (ll = 2; ll < _ + 1; ll++) {

А вот все решение, переведенное на обычный мирской JavaScript (переменным я также дал более осмысленные имена)

// Вот я, мозг размером с планету, и они просят меня написать на JavaScript…
function kevinTheNumberMentioner(max){
  var result = [];
  /* почти безвредно --> */ with(result) {
 
    // извините за все это, у моей вавилонской рыбки сегодня мигрень...
    for (candidate = 2; candidate < max + 1; candidate++) {
      var factor = 1;
      while (candidate % ++factor);
      // У меня в правом полушарии все болит от этих чертовых точек с запятой
      (candidate == factor) && push(candidate);
    }
    forEach(alert);
  }
 
  // ну так вы же точно делать не будете...
  return '42';
}

Славно, теперь перед нами, по крайней мере, узнаваемый JavaScript, но в нем по-прежнему таятся кое-какие странности.
Инструкция with относится к наиболее предосудительным с точки зрения «блюстителей чистоты JavaScript», но все-таки в строке 3 — именно она. JavaScript попытается заключить все свойства, на которые не указывают ссылки, в блоке with заданного объекта. Следовательно, неприкаянные методы массива push and forEach окажутся в области видимости result .

Еще одна любопытная инструкция — цикл while в девятой строке. У этого цикла нет тела, поэтому factor просто продолжает увеличиваться на единицу, пока не станет нацело делиться, давая candidate в качестве частного. В следующей строке проверяется, равны ли теперь значения candidate и factor . В таком случае, у числа нет меньших делителей, следовательно, оно простое и добавляется к result .
В строке 13 перебираем результаты и во всеуслышание объявляем каждое простое число в виде alert . Наконец, программа возвращает 42.

9dce5845482544549086c487d6dd92cf.png

4. Чарльз Диккенс

https://github.com/angus-c/literary.js/tree/master/book/dickens/prime.js

function MrsPrimmerwicksProgeny(MaxwellNumberby) {
  Number.prototype.isAPrimmerwick = function() {
    for (var AddableChopper = 2; AddableChopper <= this; AddableChopper++) {
      var BittyRemnant = this % AddableChopper;
      if (BittyRemnant == 0 && this != AddableChopper) {
        return console.log(
          'It is a composite. The dear, gentle, patient, noble', +this, 'is a composite'),
          false;
      }
    }
    return console.log(
      'Oh', +this, +this, +this, 'what a happy day this is for you and me!'),
      true;
  }
 
  var VenerableHeap = [];
  for (var AveryNumberby = 2; AveryNumberby <= MaxwellNumberby; AveryNumberby++) {
    if (AveryNumberby.isAPrimmerwick()) {
      VenerableHeap.push(AveryNumberby);
    }
  }
  return VenerableHeap;
}

Что, если бы вы могли спросить у числа, простое ли оно:

6..isPrime(); // ложь
7..isPrime(); // истина

Что бы сделал Чарльз Диккенс? Расширил бы Number.prototype . Его собственное расширение называется isAPrimmerwick (на самом деле, у всех объектов здесь причудливые диккенсовские имена) и определяется в строках 2–14. В строках 17–21 мы просто спрашиваем каждое число, простое ли оно, и добавляем простые числа в массив с результатами, который называется VenerableHeap .
Логика метода isAPrimmerwick в основном проста. Делим имеющееся число на все возможные делители. Если то или иное число делится без остатка, то является составным, в противном случае — простым.

В каждой инструкции возврата (строки 6 и 11) есть пара любопытных моментов. Во-первых, поскольку число вызывает метод в его же прототипе, на него можно сослаться при помощи this (но с префиксом +, чтобы принудительно привести его от числового объекта к примитиву). Во-вторых, Диккенс использует оператор-запятую, чтобы одновременно вызвать console.log и вернуть булевское значение.

d9d3c391947e44b288e03b01d929e562.jpg

5. Дэвид Фостер Уоллес

https://github.com/angus-c/literary.js/tree/master/book/wallace/prime.js

var yearOfTheLighteningQuickAtkinSieve = function(tops) {
  //B.P. #40 07-14
  //ELEPHANT BUTTE, NM
  var NSRS/*[1]*/ = [0,0,2,3];
  /* два конкурентных цикла мобилизуются так, что переменные i и j (каждая с
  исходным значением 1) увеличиваются на 1 при каждом инкременте 
 (правда, во вложенном виде). */
  for(var i = 1; i < Math.sqrt(tops); i++){
    for(var j = 1; j < Math.sqrt(tops); j++){
      if (i*i + j*j >= tops) {
        break;
      }
      /* Две переменные (т.e. i и j) подставляются в первую квадратичную функцию quadratic,
     и результат ее присваивается дополнительной переменной (n). */
      var n = 4*i*i + j*j;
      /* Если дополнительная переменная (т.e. n) деленная на 12, даст в остатке
      1 или 5, то у значения с этим индексом (т.e. у n) меняется знак [2]. */
      if(n <= tops && (n%12 == 1 || n%12 == 5)){
        NSRS[n] = NSRS[n] ? 0 : n;
      }
      /* Теперь мы (т.e. JavaScript) подходим ко второй квадратичной функции, and again the result
     и результат вновь присваивается (существующей) переменной n. */
      n = 3*i*i + j*j;
      /* Хотя переменная (т.e. n) вновь делится на 12, на сей раз остаток
      сравнивается с 7 для определения того, должен ли у значения с данным 
  индексом(т.e. у n) 
      меняться знак */
      if(n <= tops && (n % 12 == 7)){
        NSRS[n] = NSRS[n] ? 0 : n;
      }
      /* Теперь вы (т.e. читатель), испытываете чувство нерешительности и раскаяния,
 тем не менее, мы (т.e. JavaScript) еще не закончили. Как и следовало ожидать, теперь в ход
      идет третья квадратичная функция и(не менее предсказуемо) ее значение присваивается (уже
      уже поистрепавшейся) переменной n. */
      n = 3*i*i - j*j;
      /* Единственный интересный момент в третьей операции деления (однако и самый 
      удручающий) заключается в том, что она происходит лишь тогда, когда первая переменная
 в цикле (i) оказывается больше
      т.e. не меньше (или равна) второй переменной цикла (j) [3]. */
      if (i>j) {
        if((n <= tops) && (n % 12 == 11)){
          NSRS[n] = NSRS[n] ? 0 : n;
        }
      }
    }
  }
  /* В полуобморочном состоянии (но не доверяя фильтру кольцевой факторизации) мы
  (т.e. JavaScript) теперь предварительно определяем как составные
 все без исключения простые множители, без учета их актуального статуса: простые ли они
(т.е не составные) или составные (т.е. не простые)
*/
  for(i = 5; i < Math.sqrt(tops); i++){
    if(NSRS[i] == 1){
      for(j = i*i; j < tops; j += i*i){
        NSRS[j] = 0;
      }
    }
  }
  return NSRS.filter(Number); // [4]
}
/*
[1] Числовая система хранения и поиска информации.
[2] То есть значения, соответствующие текущему индексу [a] устанавливаются в 0, а значения 0 устанавливаются в текущие индексы.
[3] В противном случае каждый релевантный индекс [a] менял бы знак дважды.
[4] `Array.prototype.filter` будучи функцией высшего порядка, определяется в стандарте EcmaScript-262 (5-я
версия) [b]. Поскольку `Number` - это встроенная в язык функция, преобразующая любое значение в число. а Array.prototype.filter
отклоняет ложные (т.e. неистинные) значения, значения 0, будучи ложными (т.e. неистинными) не
будут включаться в массив, возвращаемый `Array.prototype.filter`.
 
[a] т.e. индекс, на котором рассматриваемая квадратичная функция результирует в true.
[b] http://es5.github.io/#x15.4.4.20
*/

Благодаря пространным комментариям, которыми так известен Уоллес, рассказывать особенно нечего — надо только отметить, что в основе его решения лежит сильно оптимизированное (и слишком сложное, чтобы вдаваться здесь в объяснения) решето Эткина.

Код особенно интересен изысканной логикой и уоллесовским точным, но непринужденным стилем. Однако в строке 54 есть и интересный оборот JavaScript:

return NSRS.filter(Number); // [4]

Результат — NSRS . Здесь это разрежённый массив, содержащий все простые числа, которые, однако, перемежаются с неопределенными значениями (спереди он заполнен нулями):

[0, 0, 2, 3, undefined, 5, undefined, 7/*, etc.. */]

Функция Array.prototype.filter создает новый массив, в котором содержатся лишь те элементы исходного массива, для которых заданная функция возвращает значение true . Здесь речь идет о Number, встроенной в язык функции, которая пытается принудительно привести свой аргумент к числу. Number принудительно приводит undefined к NaN, оставляя все настоящие числа нетронутыми. Поскольку оба значения: NaN и 0 означают «ложь», вновом массиве будут содержаться только простые числа:

[0, 0, 2, 3, undefined, 5, undefined, 7].filter(Number); //[2, 3, 5, 7]

Заключение

Вот и все. Надеюсь, вам понравилось.

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

© Habrahabr.ru