Подтвердить что ты не робот

Как работает этот код?

Я наткнулся на этот код на Quora.

#include<stdio.h>

main(){
          int           $[]
        ={0x69,       0154,107,
     'e',0x79,0157, 117,'v',0x6a}
     ,_,__;_=__^__;__=_;while(_<(-
      (~(1<<3))+3)){(_==1<<1||_==
       -(~(1<<3))||_==11)?putchar
        (*($+(1>>1))):putchar(*(
          __++ +$)),(_==1>>1||
            _==1<<2||_==(1<<3
              )-1)?putchar
                (' '):1;
                  _++;
                    }
}

Вывод программы i like you viji. Это трогательно, но загадочно. Поэтому я отформатировал его indent, чтобы получить лучшую идею.

main ()
{
  int $[] = { 0x69, 0154, 107,
    'e', 0x79, 0157, 117, 'v', 0x6a
  }
  , _, __;
  _ = __ ^ __;
  __ = _;
  while (_ < (-(~(1 << 3)) + 3))
    {
      (_ == 1 << 1 || _ ==
       -(~(1 << 3)) || _ == 11) ? putchar
(*($ + (1 >> 1))) : putchar (*(__++ + $)), (_ == 1 >> 1 ||
                                _ == 1 << 2 || _ == (1 << 3) - 1) ? putchar
(' ') : 1;
      _++;
    }
}

Теперь это не так трогательно, но все же немного загадочно.

Итак, может кто-нибудь объяснить, как этот код удается распечатать i like you viji?

UPDATE:

дал лучшие имена переменным $, _ и __ и расширил тройные операторы:

int a[] = { 0x69, 0154, 107, 'e', 0x79, 0157, 117, 'v', 0x6a }, x, y;

x = y ^ y;
y = x;

while (x < (-(~(1 << 3)) + 3))
  {
    if (x == 1 << 1 || x == -(~(1 << 3)) || x == 11)
      putchar (*(a + (1 >> 1)));
    else
      {
        putchar (*(y++ + a));
        if (x == 1 >> 1 || x == 1 << 2 || x == (1 << 3) - 1)
          putchar (' ');
        else
          1;
      }
    x++;
  }
4b9b3361

Ответ 1

Переписывание кода дает:

int arr1[] = { 'i', 'l', 'k', 'e', 'y', 'o', 'u', 'v', 'j'};
int i0, i1;  // Moved to new line instead of using ,

i0 = 0;  // i0 = i1 ^ i1;
i1 = i0;

while (i0 < 12)  // All strange constant like (1<<3) recalculated
{
    if (i0 == 2 || i0 == 9 || i0 == 11)   // "? :" replaced by if 
    {
            putchar(*arr1);
    }
    else
    {
        putchar (*(arr1 + i1));
        ++i1;

        if (i0 == 0 || i0 == 4 || i0 == 7)
        {
            putchar(' ');
        }
    }
    i0++;
}

Пошаговое описание:

1) Переформатируйте линии, удалите ненужные пробелы, вставьте пробелы для удобочитаемости

main()
{
    int $[] = {0x69, 0154,107, 'e',0x79,0157, 117,'v',0x6a} , _, __;
    _ = __^__;
    __ = _;
    while(_ < (-(~(1<<3))+3))
    {
        (_ == 1<<1 || _ == -(~(1<<3)) || _ == 11) ?
                putchar(*($ + (1>>1))) :
                putchar(*(__++ +$)), (_ == 1 >> 1 || _ == 1<<2 || _ == (1<<3)-1) ?
                        putchar(' ') : 1;
        _++;
    }
}

2) Переименуйте переменную, i.e $в arr1, _ в i0 и __ в i1

main()
{
    int arr1[] = {0x69, 0154,107, 'e',0x79,0157, 117,'v',0x6a} , i0, i1;
    i0 = i1^i1;
    i1 = i0;
    while(i0 < (-(~(1<<3))+3))
    {
        (i0==1<<1 || i0== -(~(1<<3)) || i0 == 11) ?
                putchar(*(arr1+(1>>1))) :
                putchar(*(i1++ +arr1)), (i0 == 1 >> 1 || i0 == 1<<2 || i0 == (1<<3)-1) ?
                        putchar(' ') : 1;
        i0++;
    }
}

3) Используйте if-statement вместо?: Это включает разбиение запятой на две строки.

main()
{
    int arr1[] = {0x69, 0154,107, 'e',0x79,0157, 117,'v',0x6a} , i0, i1;
    i0=i1^i1;
    i1=i0;
    while(i0 < (-(~(1<<3))+3))
    {
        if (i0 == 1<<1 ||i0== -(~(1<<3)) || i0 == 11)
        {
                putchar(*(arr1+(1>>1)));
        }
        else
        {
                putchar(*(i1++ +arr1));
                if (i0 == 1 >> 1 || i0 == 1<<2 || i0 == (1<<3)-1)
                {
                        putchar(' ');
                }
                else
                {
                    1;  // This does nothing so it can be removed
                }
        }
        i0++;
    }
}

4) Пересчитать числовые константы на лучшие значения Примеры

0x69 совпадает с "i"

1 < 1 совпадает с 2

- (~ (1 < 3)) является таким же, как 9

i1 ^ i1 совпадает с 0

1 → 1 совпадает с 0

main()
{
    int arr1[] = { 'i', 'l', 'k', 'e', 'y', 'o', 'u', 'v', 'j'} , i0, i1;
    i0 = 0;
    i1 = i0;
    while(i0 < 12)
    {
        if (i0 == 2 || i0 == 9 || i0 == 11)
        {
                putchar(*(arr1));
        }
        else
        {
                putchar(*(i1++ +arr1));

                if (i0 == 0 || i0 == 4 || i0 == 7)
                {
                        putchar(' ');
                }
        }
        i0++;
    }
}

5) Небольшая финальная очистка

main()
{
    int arr1[] = { 'i', 'l', 'k', 'e', 'y', 'o', 'u', 'v', 'j'};  // Move i0 and
                                                                  // i1 to nextt line
    int i0, i1;

    i0 = 0;
    i1 = i0;
    while(i0 < 12)
    {
        if (i0 == 2 || i0 == 9 || i0 == 11)
        {
                putchar(*arr1);
        }
        else
        {
                putchar(*(arr1 + i1));  // Splitted into two lines
                ++i1;

                if (i0 == 0 || i0 == 4 || i0 == 7)
                {
                        putchar(' ');
                }
        }
        i0++;
    }
}

Теперь код довольно легко читается.