Программа подписи - Signature program

Пример программы подписи, скомпилированной и запущенной в Windows CLI.

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

Программы подписи обычно находятся в разделе подписей сообщений пользователей на форумах, в сообщениях и т. Д., Особенно на технически ориентированных веб-сайтах, таких как Slashdot.

Оптимизация и пример

Для того, чтобы программа, часто очень сложная, которая обычно была бы представлена ​​множеством строк кода, помещалась в одну строку, используется множество методов сжатия. Следующая программа распечатывает ASCII искусство версия Треугольник Серпинского фрактал, показанный на иллюстрации:

главный(c,р){за(р=32;р;) printf(++c>31?c=!р--,"":c<р?" ":~c&р?" `":" #");}

Несжатая «прописанная» версия этой программы может выглядеть так:

#включают <stdio.h>int главный(пустота) {    int c = 0;    int р = 32;    пока (р != 0) {        c = c + 1;        если (c > 31) {            c = !р;   / * то же, что c = 0 * /            р = р - 1;            printf("");        }        еще {            если (c < р) {                printf(" ");            }            еще {                если ((~c & р) != 0) {                    printf(" `");                }                еще {                    printf(" #");                }            }        }    }    возвращаться 0;}

Здесь мы видим, что многие части кода отсутствуют в программе подписи, наиболее очевидно отсутствие #include , нет возвращаемого типа главный() а также использование нестандартных основной (c, r). В ?: Оператор играет жизненно важную роль в уменьшении общего размера программы, позволяя одному выражению заменить значительное количество если/еще логические условия. Размер также значительно уменьшается за счет удаления всех пробел и разрывы строк.

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

Обфусцированные программы

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

MinRay

Компьютерная графика эксперт Пол Хекберт вывел концепцию сжатой программы на новый уровень, разборчиво распечатав результаты конкурса за минимальную трассировка лучей программа на обороте визитная карточка. Код из Графика Самоцветы IV хранилище воспроизводится здесь.

typedef структура{двойной Икс,у,z}vec;vec U,чернить,Amb={.02,.02,.02};структура сфера{vec cen, цвет;двойной рад,kd,кс,kt,kl,ir}*s,*Лучший,sph[]={0.,6.,.5,1.,1.,1.,.9,.05,.2,.85,0.,1.7,-1.,8.,-.5,1.,.5,.2,1.,.7,.3,0.,.05,1.2,1.,8.,-.5,.1,.8,.8,1.,.3,.7,0.,0.,1.2,3.,-6.,15.,1.,.8,1.,7.,0.,0.,0.,.6,1.5,-3.,-3.,12.,.8,1.,1.,5.,0.,0.,0.,.5,1.5,};yx;двойной ты,б,tmin,sqrt(),загар();двойной вдот(А,B)vec А,B;{возвращаться А.Икс*B.Икс+А.у*B.у+А.z*B.z;}vec vcomb(а,А,B)двойной а;vec А,B;{B.Икс+=а*А.Икс;B.у+=а*А.у;B.z+=а*А.z;возвращаться B;}vec вунит(А)vec А;{возвращаться vcomb(1./sqrt(вдот(А,А)),А,чернить);}структура сфера*пересекаться(п,D)vec п,D;{Лучший=0;tmin=1e30;s=sph+5;пока(s-->sph)б=вдот(D,U=vcomb(-1.,п,s->cen)),ты=б*б-вдот(U,U)+s->рад*s->рад,ты=ты>0?sqrt(ты):1e31,ты=б-ты>1e-7?б-ты:б+ты,tmin=ты>=1e-7&&ты<tmin?Лучший=s,ты:tmin;возвращаться Лучший;}vec след(уровень,п,D)vec п,D;{двойной d,эта,е;vec N,цвет;структура сфера*s,*л;если(!уровень--)возвращаться чернить;если(s=пересекаться(п,D));еще возвращатьсяAmb;цвет=Amb;эта=s->ir;d= -вдот(D,N=вунит(vcomb(-1.,п=vcomb(tmin,D,п),s->cen)));если(d<0)N=vcomb(-1.,N,чернить),эта=1/эта,d= -d;л=sph+5;пока(л-->sph)если((е=л->kl*вдот(N,U=вунит(vcomb(-1.,п,л->cen))))>0&&пересекаться(п,U)==л)цвет=vcomb(е,л->цвет, цвет);U=s->цвет;цвет.Икс*=U.Икс;цвет.у*=U.у;цвет.z*=U.z;е=1-эта*эта*(1-d*d);возвращаться vcomb(s->kt,е>0?след(уровень,п,vcomb(эта,D,vcomb(эта*d-sqrt(е),N,чернить))):чернить,vcomb(s->кс,след(уровень,п,vcomb(2*d,N,D)),vcomb(s->kd,цвет,vcomb(s->kl,U,чернить))));}главный(){printf("% d% d",32,32);пока(yx<32*32)U.Икс=yx%32-32/2,U.z=32/2-yx++/32,U.у=32/2/загар(25/114.5915590261),U=vcomb(255.,след(3,чернить,вунит(U)),чернить),printf("% .0f% .0f% .0f",U);}/ * минрей! * /

Хотя для подписи это неудобно, но для трассировщика лучей он удивительно короткий (хотя и неэффективный).

Смотрите также

внешняя ссылка