Як працює цей заплутаний JavaScript?


93

Як працює наступний JavaScript?

Я розумію, що це мінімізований код. Я спробував трохи знеструмити це, але не можу отримати чітке уявлення про те, як він досягає цього ефекту. Я бачу, що він використовує Strings для якоїсь ітерації, використання об'єкта Date, дивної маніпуляції з рядками, функцій Math, потім код друкує сам.

Як міг би бути переписаний той самий ефект мінімальним прикладом?

eval(z='p="<"+"pre>"/* ,.oq#+     ,._, */;for(y in n="zw24l6k\
4e3t4jnt4qj24xh2 x/* =<,m#F^    A W###q. */42kty24wrt413n243n\
9h243pdxt41csb yz/* #K       q##H######Am */43iyb6k43pk7243nm\
r24".split(4)){/* dP      cpq#q##########b, */for(a in t=pars\
eInt(n[y],36)+/*         p##@###YG=[#######y */(e=x=r=[]))for\
(r=!r,i=0;t[a/*         d#qg `*PWo##q#######D */]>i;i+=.05)wi\
th(Math)x-= /*        aem1k.com Q###KWR#### W[ */.05,0>cos(o=\
new Date/1e3/*      .Q#########Md#.###OP  A@ , */+x/PI)&&(e[~\
~(32*sin(o)*/* ,    (W#####Xx######.P^     T % */sin(.5+y/7))\
+60] =-~ r);/* #y    `^TqW####P###BP           */for(x=0;122>\
x;)p+="   *#"/* b.        OQ####x#K           */[e[x++]+e[x++\
]]||(S=("eval"/* l         `X#####D  ,       */+"(z=\'"+z.spl\
it(B = "\\\\")./*           G####B" #       */join(B+B).split\
(Q="\'").join(B+Q/*          VQBP`        */)+Q+")//m1k")[x/2\
+61*y-1]).fontcolor/*         TP         */(/\\w/.test(S)&&"#\
03B");document.body.innerHTML=p+=B+"\\n"}setTimeout(z)')//

JSFiddle


8
Класна анімація ... може закінчитися використанням цього десь насправді!
tymeJV

7
О, гарно. Не помітив загадку.
ThiefMaster

37
Це називається Quine, і це одна з найбільш фантастичних Quine's, яку я коли-небудь бачив. en.wikipedia.org/wiki/Quine_(computing)
Девід Сутер

9
@Roko C. Buljan Я думаю, що це його сторінка: aem1k.com
Олександр

5
Схоже, автор зараз розмістив помічену версію на GitHub.
Der Hochstapler

Відповіді:


67

Вступне слово : Я широко прикрасив та анотував код на http://jsfiddle.net/WZXYr/2/

Розглянемо самий зовнішній шар:

eval(z = '...');

Рядок коду зберігається у змінній z. Оператор присвоєння повертає призначене значення, тому рядок коду також передається як аргумент eval.

Рядок коду zпрацює всередині eval. Код є надзвичайно тупим, навіть при очищенні, але, схоже, це:

  1. Розбираємо рядок чисел базових 36, окреслених символом 4.
  2. Заповнюємо карту значень, використовуючи глобальні змінні e, xі yдля утримання карти стану. Стан карти є частково функцією поточної секунди на настінному годиннику ( new Date / 1e3).
  3. Використовуючи значення карт, код генерує вихідний рядок, p
    • код використовує, p += " *#"[index]щоб визначити, чи використовувати пробіл, зірочку чи хеш-позначку, де indexє насправді e[x++] + e[x++](як сказано вище, eі xвідповідають за стан карти)
    • якщо індекс більший, ніж довжина " *#", є резервний код, який заповнює висновок pсимволами з z. Внутрішні символи заповнені анімаційними персонажами, а зовнішні символи - з них z.

В кінці коду є виклик до setTimeout(z), який асинхронно оцінює рядок коду z. Це повторне виклик zдозволяє коду циклічно.

Простий приклад:

Ось надзвичайно проста версія ( http://jsfiddle.net/5QXn8/ ):

eval(z='p="<"+"pre>";for(i=0;i<172;++i)if(i > 62 && i < 67)p+="!---"[~~(new Date/1e2 + i)%4];else p += ("eval(z=\'" + z + "\')")[i];document.body.innerHTML = p;setTimeout(z)')
  1. forЦикл додає кожен символ у вихідний рядок p(рядок 172 символів):

    for(i=0;i<172;++i)
  2. Внутрішній умовний вирішує, чи ми знаходимося на персонажі між позицією 62 до 67, які є анімованими символами:

    if(i > 62 && i < 67)
  3. Якщо ми є, то роздруковуємо !---зсув, виходячи з десятої частини другого значення настінного годинника. Це забезпечує ефект анімації.

    p+="!---"[~~(new Date/1e2 + i)%4]

    (Усі гнучкість навколо new Dateнасправді є лише для того, щоб перетворити значення дати в число від 0 до 3.)

  4. В іншому випадку, якщо ми не анімовані, надрукуйте iсимвол- індекс із рядка, визначеного символом

    "eval(z='" + z + "')"

    Тобто рядок коду, zоточений eval('і ').

  5. Нарешті, виведіть рядок та використовуйте setTimeoutдля черги в іншому виконанні z:

    document.body.innerHTML = p;setTimeout(z)

Зауважте, що мій підсумковий результат не зовсім правильний - я не рахував зворотних нахилів до кінця, - але він все одно повинен дати вам досить гарне уявлення про те, як працює техніка в цілому.


8
Зверніть увагу на цей github.com/aemkei/world/blob/master/annotated.js - власну коментовану версію автора в GitHub.
Бенджамін Груенбаум

36

Ось анотоване джерело. Ps: я автор;)

function z(){                     // will be replaced with eval

  p = "<" + "pre>";               // use <pre> tag for formatted output

  for (                           // loop though lines
    y in n = (                    // y - the line number
      "zw24"      +               // n - the encoded data
      "l6k4"      +               // every line holds encoded data
      "e3t4"      +
      "jnt4"      +               // string will be concated in build process
      "qj24"      +
      "xh2  4"    +               // data after spaces will be ignored but
      "2kty24"    +               // … is used to not break block comments
      "wrt4"      +               // … which will save some chars
      "13n24"     +
      "3n9h24"    +
      "3pdxt4"    +
      "1csb   4"  +
      "3iyb6k4"   +
      "3pk724"    +
      "3nmr24"
    ).split(4)                    // data will be split by (unused) 4

  ){
    for (                         // loop throug every char in line
      a in t = parseInt(          // numbers are encoded as string
        n[y],                     // … with a base of 36
        36
      ) + (                       // large number will be converted to string
        e =                       // e - holds the rendered globe
        x =                       // x - horizonal position
        r = []                    // r - bitmap flag if pixel is set
      )
    ){
      r = !r;                     // toggle binary flag

      for (                       // look though bitmap states
        i = 0;                 
        t[a] > i;                 // draw pixel t[a]-times
        i += .05
      )
        with (Math)               // refer to Math later
          x -= .05,
          0 > cos(                // prevent backface visibility
            o =
              new Date / 1e3 +    // get rotation based on current time
              x / PI
          ) && (
            e[                    // access matrix
              ~~(                 // convert float to integer
                sin(o) *          // rotate around y axis
                sin(.5 + y/7) *
                32                // scale up the globe
              ) + 60              // move to center
            ] = -~r               // store bitmap state in render matrix
          )
    }

    for (                         // loop through columns
      x = 0;
      122 > x;                    // break after char 122
    ) p += "   *#"[               // add space, asterisk or hash
        e[x++] +                  // … based pixel opacity
        e[x++]
      ] || (S = (                 // otherwise use the original code
        "eval(z='" +              // inception of missing "eval" statement
          z
            .split(B = "\\")      // escape \ with \\
            .join(B + B)

            .split(Q = "'")       // escape ' with \'
            .join(B + Q) +

          Q +                     // add missing ')

          ")////////"             // add extra chars to fill mapping
        )[
          x / 2 +                 // get character at current position
          61 * y-1
        ]

      ).fontcolor(                // colorize outpu
        /\w/.test(S) &&           // test for [0-9A-Z]
        "#03B"                    // render blue
                                  // otherwise pink (default)
      );

    document.body.innerHTML =     // render output
      p +=                        // append new line
      B +                         // add backspace
      "\n";                       // add new line
  }

  setTimeout(z)                   // render animation on next frame
}
z()

5
Зауважте, це також пояснено у цьому відео youtube.com/watch?v=RTxtiLp1C8Y
Бенджамін Груенбаум

21

Ось ще одна знезаражена версія вручну, яка переміщує всю ініціалізацію з виразу у власні заяви:

z='p="<"+"pre>"/* ,.oq#+     ,._, */;for(y in n="zw24l6k\
4e3t4jnt4qj24xh2 x/* =<,m#F^    A W###q. */42kty24wrt413n243n\
9h243pdxt41csb yz/* #K       q##H######Am */43iyb6k43pk7243nm\
r24".split(4)){/* dP      cpq#q##########b, */for(a in t=pars\
eInt(n[y],36)+/*         p##@###YG=[#######y */(e=x=r=[]))for\
(r=!r,i=0;t[a/*         d#qg `*PWo##q#######D */]>i;i+=.05)wi\
th(Math)x-= /*        aem1k.com Q###KWR#### W[ */.05,0>cos(o=\
new Date/1e3/*      .Q#########Md#.###OP  A@ , */+x/PI)&&(e[~\
~(32*sin(o)*/* ,    (W#####Xx######.P^     T % */sin(.5+y/7))\
+60] =-~ r);/* #y    `^TqW####P###BP           */for(x=0;122>\
x;)p+="   *#"/* b.        OQ####x#K           */[e[x++]+e[x++\
]]||(S=("eval"/* l         `X#####D  ,       */+"(z=\'"+z.spl\
it(B = "\\\\")./*           G####B" #       */join(B+B).split\
(Q="\'").join(B+Q/*          VQBP`        */)+Q+")//m1k")[x/2\
+61*y-1]).fontcolor/*         TP         */(/\\w/.test(S)&&"#\
03B");document.body.innerHTML=p+=B+"\\n"}setTimeout(z)';

p = "<" + "pre>";
n = ["zw2", "l6k", "e3t", "jnt", "qj2", "xh2 x/* =<,m#F^    A W###q. */", "2kty2", "wrt", "13n2", "3n9h2", "3pdxt", "1csb yz/* #K       q##H######Am */", "3iyb6k", "3pk72", "3nmr2", ""]
for (y in n) {
    e = [];
    x = 0;
    r = true;
    t = parseInt(n[y], 36) + "";
    for (a in t) {
        r = !r
        for (i = 0; i < t[a]; i += 0.05) {
             x -= 0.05;
             o = new Date / 1e3 + x / Math.PI
             if (Math.cos(o) < 0)
                 e[~~(32 * Math.sin(o) * Math.sin(0.5 + y / 7)) + 60] = -~r;
        }
    for (x = 0; x < 122;) {
        S = "eval" + "(z='" + z.split(B = "\\").join(B + B).split(Q = "'").join(B + Q) + Q + ")//m1k"
        p += "   *#"[e[x++] + e[x++]] || S[x/2+61*y-1]).fontcolor(/\w/.test(S[x/2+61*y-1]) && "#03B");
    }
    p += B + "\n";
    document.body.innerHTML = p;
}
setTimeout(z)

Ось що відбувається:

  • zявляє собою багаторядковий рядок, що містить весь код. Це evalред.
  • В кінці коду zпередається в setTimeout. Він працює як requestAnimationFrameі evalразом, оцінюючи його в інтервалі з максимальною швидкістю.
  • Сам код ініціалізує p, буфер рядків, до якого буде доданий HTML, і nмасив номерів, закодованих базовою 36, (об'єднані в рядок, при "4"цьому коментарі не мають значення сміття, яке не враховується parseInt).
  • кожне число в nкодує один рядок ( n.length == 16). Зараз це перелічено .
  • Ініціалізується купа змінних, деякі маскуються під eлітерал масиву, але потім вони передаються цифрам ( x) або booleans ( r) або рядкам ( t) при використанні.
  • Кожна цифра в цифрі tперераховується, перевертаючи булеве значення rкожного витка. Для різних ракурсів xі залежно від поточного часу new Date / 1000 (щоб він давав анімацію), масив eзаповнюється за допомогою деяких побітових операторів - з 1коли rнеправда, а 2з коли r- істинно в той час.
  • Потім цикл робить ітерацію 61 стовпців зображення, від x=0122 до подвійних кроків, додаючи до символів окремі символи p.
  • Bбудучи зворотною косою рисою, рядок Sбудується з рядка коду zшляхом уникнення косої коси та апострофи, щоб отримати точне уявлення про те, що воно виглядало в джерелі.
  • Кожні два послідовні номери eдодаються та використовуються для доступу до персонажа " *#", для створення анімованого зображення. Якщо один з індексів не визначений, NaNіндекс переходить на невизначений символ, а замість нього Sбереться відповідний символ із рядка (ознайомтеся з формулою x/2+61*y-1). Якщо цей символ повинен бути символом слова , він забарвлюється по-різному за допомогою fontcolorметоду String .
  • Після кожного рядка додається зворотний простір pта рядок рядків , а рядок HTML присвоюється тілу документа.

Як такий же ефект можна було б переписати на мінімальний приклад?

Ось інший приклад:

setInterval(z='s=("setInterval(z=\'"+\
z.replace(/[\\\\\']/g,"\\\\$&")+"\')"\
).match(/.{1,37}/g).join("\\\\\\n");d\
ocument.body.innerHTML=\"<\\pre>"+s.s\
lice(0, 175)+String( + new Date()).fo\
ntcolor("red")+s.slice(188)')

( демонстрація на jsfiddle.net )

У ньому є всі необхідні речі, необхідні для такого виду анімації:

  • setIntervalі Dateдля анімації
  • Реконструкція власного коду ( схожий на quine) тут:

    s = ( "setInterval(z='" // the outer invokation
          + z.replace(/[\\\']/g,"\\$&") // the escaped version
        + "\')" ) // the end of the assignment
        .match(/.{1,37}/g).join("\\\n"); // chunked into lines
  • Вихід через document.body.innerHTMLі <pre>елемент

  • Заміна деяких частин коду анімованим рядком

2
треба визнати, чудова відповідь!
rafaelcastrocouto

5

Обчислюється рядок із усім кодом, і цикл робить тайм-аут; Рядок зберігається у змінній з назвою zта посеред коду, між коментарями, /*і */є "Art ASCII Art" Землі. Код аналізує коментарі та змінює вміст документа, зберігаючи js та оновлюючи зображення. Нижче наведено лише нарізаний код:

  p="<pre>";
  for(y in n="zw24l6k4e3t4jnt4qj24xh2 x42kty24wrt413n243n9h243pdxt41csb yz43iyb6k43pk7243nmr24".split(4)){ 
    for(a in t = parseInt(n[y],36)+(e=x=r=[]))
      for(r=!r,i=0;t[a]>i;i+=.05)
        with(Math) x-= .05,0>cos(o=new Date/1e3+x/PI)&&(e[~~(32*sin(o)*sin(.5+y/7))+60] =-~ r);
          for(x=0;122>x;) p += "   *#"[e[x++]+e[x++\]] ||
              (S=("eval"+"(z=\'"+z.split(B = "\\\\").join(B+B).split(Q="\'").join(B+Q)+Q+")//m1k")[x/2+61*y-1]).fontcolor(/\\w/.test(S)&&"#\03B");
    p += B+"\\n"
    document.body.innerHTML= p
  }

6
Будь-який спосіб, неймовірний, як навколо екватора графіка має більше спіну ... дивовижно. +1 BTW
Roko C. Buljan
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.