Суббота, 18.05.2024
Королевство Delphi
Главное меню
Статьи
Наш опрос
Нашли свой исходник?
Всего ответов: 94
Статистика
Онлайн всего: 1
Гостей: 1
Пользователей: 0
Форма входа
Главная » Статьи » Защита » Разные

Уменьшаем размер EXE в 40 раз, или Вся правда о консольных приложениях

«Пустая» форма весит около 355 КБ, и этот начальный размер увеличивается с каждой новой версией Delphi. «Пустая» программа, написанная с использованием библиотеки KOL, уменьшающей размер исполняемого файла, — 32 КБ.

«Чистое» консольное приложение имеет размер 8 КБ, поскольку отображается как процесс и, соответственно, не имеет сложных взаимодействий с Windows-окнами. То есть можно сделать так, чтобы по Ctrl+Alt+Del консоль не было видно :).

Итак, в меню Delphi выбери File>New>Other и в появившемся окне среди прочего найди пункт Console Application. Возникнет следующая заготовка:

program Project1; //название проекта

{$APPTYPE CONSOLE} //директива, указывающая на наличие консоли

uses SysUtils; //подключенные модули

begin //начало процесса
 { TODO -oUser -cConsole Main : Insert code here } //комментарий от Borland
end. //конец процесса

Ага. Это «пустое» консольное приложение. Нажми F9, чтобы запустить его. Что ты увидел? Черное окошко вроде Сеанса MS-DOS возникло и сразу исчезло. Куда оно делось? Всё дело в том, что консольное приложение — это процесс, который, как и всё на свете, когда-нибудь закончится :). Начало процесса — ключевое слово begin, а конец — end. Поскольку между ними отсутствуют какие-либо другие команды, то end (прекращение процесса) исполняется сразу после начала, и консоль исчезает. Чтобы такого не было, надо «занять» приложение каким-нибудь циклом, желательно вечным ;). Вот так:

begin
 repeat
 //это наш вечный цикл
 until 1=0;
end.

Обрати внимание на команду until. Наш цикл будет исполнятся до тех пор, пока 1 не станет равен 0. Угадайте сами, когда это случится :). Другой вариант:

begin
 while true do begin
 //вставляй код здесь
 end;
end.

И еще вариант:

Label MyLabel; //«метка»
begin
 MyLabel:
 //твой код здесь
 goto MyLabel;
end.

В общем, вариантов сколько угодно. Главное, что консоль не будет закрываться. А теперь надо реализовать чтение и запись на полотно консоли, как это сделано в (не)старом (не)добром MS-DOS. Помогут нам в этом процедуры из модуля System.pas. Синтаксис:

WriteLn(ЧТО_ЗАПИСЫВАЕМ) //вывод данных в консоль

ReadLn(ЧТО_ЧИТАЕМ) //чтение данных из консоли

Почему же модуль System.pas не продекларирован в разделе uses? Это базовый модуль Delphi, который всегда подключен «по умолчанию». А теперь добавь к исходному коду:

WriteLn(‘Hello World!’);

Соответствующая строка («Hello World!») будет выведена на консоль. Если эта команда будет помещена в вечный цикл (как его создать — см. выше), то строка "Hello World!” тоже будет добавляться бесконечное число раз. Чтобы это исправить, нужно написать:

Begin
 While true do begin
 Writeln(‘Hello World!’);
 Readln;
 End;
End.

Как ты уже понял, команда WriteLn записывает, а ReadLn — читает. При этом команда, стоящая после ReadLn, выполнится, только когда юзер нажмет клавишу Enter.

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

var S: String; //наша переменная
begin
while true do begin
Writeln(‘Enter your name’+#10);
Readln(S);
Writeln(#10+‘Your name is ’+S);
end;
end.

Здесь «№10» обозначает конец абзаца, переход курсора на следующую строку (клавиша Enter). А вот пример, где программа закрывается по команде юзера:

var s: String;

begin
while true do begin
Readln(S); //что ввел юзер?
//Юзер мог ввести команду и прописными буквами,
//и строчными. Преобразуем буквы в прописные
//командой UpperCase.
If UpperCase(s)=‘EXIT’ then begin
//переспросим еще раз
Writeln(‘Do you really want to exit? [y/n]’);
//читаем ответ юзера
Readln(s);
if UpperCase(s)=’Y’ then exit; //выходим
end;
end;
end.

Вот так. Сюда можно вставить какой угодно код, только подключив, если требуется, необходимые модули. Теперь еще раз откомпилируй проект и нажми Project>Information for ‘ProjectName’. Размер EXE будет около 40 килобайт, но только потому, что модуль SysUtils.pas в разделе Uses весит так много. А если ты заменишь этот модуль на Windows.pas, то программа будет занимать, как я и обещал, ВОСЕМЬ :) кило на харде :). Конечно, при условии, что ты будешь пользоваться только модулем Windows, который содержит большинство команд, необходимых в повседневности. Если ты не собираешься вступать в консольные переговоры с юзером и пользоваться процедурами WriteLn и ReadLn, то и консоль не нужна. Удали директиву {$APPTYPE CONSOLE}, чтобы черное MS-DOS’овское окошко не появлялось.

И помни: не пытайся указывать русские буквы в команде WriteLn: консоль отобразит их в другой кодировке. Чтобы это исправить, напечатай исходный (русский) текст в Блокноте и поставь шрифт Terminal. Результат будет в кодировке DOS, как его и надо указывать в процедуре WriteLn.

Очистить полотно консоли от текста можно так:

program Project1;

{$APPTYPE CONSOLE}

uses Windows;

var
buffer: TConsoleScreenBufferInfo; //буфер
i: integer;
begin
WriteLn('Press <Enter> to clear screen');
ReadLn;
GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE),buffer);
for i:=0 to buffer.dwSize.y do writeln;
Writeln('Screen is cleared :)');
Readln;
end.

Как поместить консольное приложение в StartUp? Ответ: использовать модуль ShlObj.pas.

program StartUp;

{$APPTYPE CONSOLE}

uses
 ShlObj, //!!
 SysUtils,
 Windows;
var
 Folder: Pchar; //путь к StartUp
 List: PitemidList; //список "специальных" папок
begin
 //ищем папку
 SHGetSpecialFolderLocation(0,CSIDL_STARTUP,List);
 new(folder);
 SHGetPathFromIDList(List,folder);
 //Нашли? Переходим в директорию StartUp
 ChDir(folder);
 //копируем файл
 CopyFile(PChar(ExtractFilePath(paramStr(0)) + 'StartUp.exe'), 'StartUp.exe', true); //укажите имя своего EXE файла
end.

Теперь загляни в папку «Автозагрузка». Если ты указал в функции имя СВОЕГО файла, он должен быть уже там :).

И маленькое Западло на закуску:

//эта программа меняет системные цвета :)
program Joke;
uses
 Windows;
const
 SysColorArray: array [0..13] of Integer = (COLOR_ACTIVEBORDER, COLOR_ACTIVECAPTION, 
 COLOR_APPWORKSPACE, COLOR_BACKGROUND, COLOR_BTNFACE, COLOR_BTNTEXT, COLOR_CAPTIONTEXT, 
 COLOR_INACTIVEBORDER, COLOR_INFOTEXT, COLOR_MENU, COLOR_MENUTEXT, COLOR_WINDOW, 
 COLOR_WINDOWFRAME, COLOR_WINDOWTEXT);
 ColorArray: array [0..12] of Integer = (16776960, 0, 16711680, 65535, 16711935, 32768, 
 8388608, 255, 12632256, 16777215, 15780518, 128, 32896);
 //Цвета хранятся в модуле Graphics.pas,
 //но мы не будем использовать его,
 //а запишем цвета в цифровом виде.
begin
Sleep(30000) //спим 30 секунд
SetSysColors(1, SysColorArray[random(13)], ColorArray[random(12)]);
end.

Автор работы: Трофим Роцкий

Получить ссылку на материал

Категория: Разные | Добавил: Барон (15.12.2011)
Просмотров: 1255 | Теги: размер, EXE | Рейтинг: 0.0/0
[ Пожертвования для сайта ] [ Пожаловаться на материал ]

Если вам помог материал сайта кликните по оплаченной рекламе размещенной в центре

Поиск
Категории раздела
Бомберы [0]
Трояны [0]
Робота с паролем [4]
Delphi и Хакинг [2]
Шифрование [6]
Разные [25]
Королевство Delphi © 2010-2024
Яндекс цитирования