FAQ
Меню сайта


Поиск

Календарь
«  Июль 2025  »
ПнВтСрЧтПтСбВс
 123456
78910111213
14151617181920
21222324252627
28293031


Друзья сайта

Категории раздела

Приветствую Вас, Гость · RSS 02.07.2025, 08:05

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

Список вопросов:

1)Q:> А как восстановить исходники на паскале из .exe/.tpu?

2)Q:> Как использовать TPU для 6-й версии TP в 7-й версии?

3)Q:> Хочу паскаль под DOS, но в защищенном режиме! 32-bit protected! Под Win32! > Под OS/2!

4)Q:> А почему задержки, которые задаются процедурой Delay(),на разных машинах > разные?

5)Q:> А чем тогда делать фиксированную задержку, вместо Delay()?

6)Q:> А как убрать курсор в текстовом режиме?

7)Q:> А как его потом обратно включить?

8)Q:> А можно сделать так, чтобы курсор был виден, но не мигал?

9)Q:> А почему программы, написанные на паскале, вылетают с ошибкой > Runtime Error 200 (Division by zero) на быстрых машинах типа PII-300?

10)Q:> Почему глючат процедуры побитового сдвига shl, shr > в применении к LongInt?

11)Q:> Почему при умножении 2-х чисел получается неверный результат:


 var a,b:Word;
 l:LongInt;
 a:=1234;
 b:=567;
 l:=a*b;
 
В l получается. 44318, а не 699678, как должно быть на самом деле. Это баг?

12)Q:> А как из моей программы запустить другую?

13)Q:> А почему внешняя программа из моей не запускается?

14)Q:> А если мне самому память нужна?

15)Q:> Как сделать процедуру с нефиксированным числом параметров, типа writeln?

16)Q:> Как с этим боpоться (сделать одну пpоцедуpу)?

17)Q:> Как узнать ErrorLevel,который выставила программа, запущенная через Exec()?

18)Q:> Предположим, запустили мой ЕХЕ. > Как узнать из программы, в каком каталоге он лежит, > и не переименовали ли его?

19)Q:> Как узнать, какие параметры передаются моему ЕХЕ при вызове из > командной строки?

20)Q:> А можно записать в середину текстового файл строку произвольной длины, > но чтобы остальные строки остались, как были, или удалить одну строку из > середины?

1)Q:> А как восстановить исходники на паскале из .exe/.tpu?

1)A: Hикак! Хотя, если есть желание помучиться, то... В пaкете Turbo Analyst 7.0 есть утилиткa, котоpaя может восстaновить интеpфейсную чaсть скомпилиpовaнного модуля, a тaкже изaссемблиpовaть implementation-чaсть.

2)Q:> Как использовать TPU для 6-й версии TP в 7-й версии?
2)A: Если нет исходников, то Hикак!

3)Q:> Хочу паскаль под DOS, но в защищенном режиме! 32-bit protected! Под Win32! Под OS/2!
3)A: В Borland Pascal 7.0 можно использовать только_ 16-битный защищенный режим,он же DPMI. (Turbo Pascal - это всего лишь облегченная версия BP, работающая только в реальном режиме, посему запустите BP.exe и узнаете много нового (хотя бы посмотрите на возможности пункта TARGET при компиляции).Для 32-bit защищенного режима существуют более лругие компиляторы:

4)Q:> А почему задержки, которые задаются процедурой Delay(),на разных машинах разные?
4)A: Юнит CRT криво написан.

5)Q:> А чем тогда делать фиксированную задержку, вместо Delay()?
5)A: Если достаточна точность в 1 тик (1/18 доля секунды), то так:


 Procedure Delay(x:longint);
 {пауза на Х тиков}
 var
 l:longint;
 begin
 l := MemL[Seg0040:$6c];
 while MemL[Seg0040:$6c] < l+x do; {задержка на X 
 тиков}
 end;
 Если нужно точнее, то надо читать регистры таймера...
 Вот вам другая процедура:
 procedure RTCDelay(Wait:LongInt); assembler;
 {Ожидание на Wait микросекунд }
 asm
 mov ah,86h
 mov dx,word ptr Wait
 mov cx,word ptr Wait+2
 int 15h
 end;
 
Wait - в *микро*секундах, но разрешение все равно с точностью до ~976 микросекунд (примерно 1 миллисекунда), т.е. меньше 1024 ставить просто бессмысленно.

6)Q:> А как убрать курсор в текстовом режиме?
6)A: procedure CursorOff; assembler;

 asm
 mov ah,1
 mov cx,2020h {Убрать мерцание за пределы знакоместа}
 int 10h
 end;
 

7)Q:> А как его потом обратно включить?
7)A:


 procedure CursorOn; assembler; 
 asm
 mov ah,1
 mov cx,0607h {Установить мерцание 6й и 7й строк}
 int 10h
 end;
 

8)Q:> А можно сделать так, чтобы курсор был виден, но не мигал?
8)A: Hет. Только самому рисовать '_' (подчеркивание) в нужном месте.

9)Q:> А почему программы, написанные на паскале, вылетают с ошибкой Runtime Error 200 (Division by zero) на быстрых машинах типа PII-300?
9)A: Еще раз: Юнит CRT криво написан. Если в программе используется модуль CRT, то программа виснет при запуске из-за инициализации, которая потом используется процедурой Delay(). Инициализация происходит в любом случае даже если эта процедура не используется в программе. При написании своих программ используйте пропатченный CRT. Чтобы заработала уже скомпилированная программа (чужая), возьмите hex-редактор и: Ищем: B93700F7F1 Меняем на: B96E00F7F1 (увеличение в 2 раза) или на: B937009090 (убрать деление вообще) Программа _не_ должна быть запакована EXE-паковщиками.

10)Q:> Почему глючат процедуры побитового сдвига shl, shr в применении к LongInt?
10)A: Известный баг. Можно поставить фикс BP7.0 -> BP7.01, или же использовать свои процедуры. Вот пример (работают на 386+):


 function LongShl(A: LongInt; B: Byte): LongInt; assembler;
 asm
 mov cl,[B]
 db $66 {код опеpаций с 32-битными pегистpами}
 mov ax,word ptr [A]
 db $66
 shl ax,cl
 db $66
 push ax
 pop ax
 pop dx
 end;
 function LongShr(A: LongInt; B: Byte): LongInt; assembler;
 asm
 mov cl,[B]
 db $66
 mov ax,word ptr [A]
 db $66
 shr ax,cl
 db $66
 push ax
 pop ax
 pop dx
 end;

11)Q:> Почему при умножении 2-х чисел получается неверный результат:


var a,b:Word;
l:LongInt;
a:=1234;
b:=567;
l:=a*b;
В l получается. 44318, а не 699678, как должно быть на самом деле. Это баг?
11)A: Hет, это фича. Тип выражения определяется только типом входящих в него переменных, но не типом переменной, куда записывается результат. Если есть опасность переполнения, надо конвертировать тип явно: l:=LongInt(a)*b;

12)Q:> А как из моей программы запустить другую?
12)A: С помощью процедуры Exec:


 SwapVectors;
 Exec(Путь&ИмяВашейПрограммы, Доп_Параметры);
 SwapVectors;
 

13)Q:> А почему внешняя программа из моей не запускается?
13)A: А память кто будет внешней проге отдавать? По умолчанию все 640Kb отдаются твоей программе. Исправить это можно либо через Options/Memory Sizes../ High heap limit, либо директивой компилятора: {$M 4096,0,10000} Здесь первая цифра - память под стек (в стеке размещаются локальные переменные каждой вызываемой процедуры/функции), вторая - нижняя граница памяти, третья - верхняя. В данном примере твоей программе отдается 10000 байт, а все остальное - внешней программе.

14)Q:> А если мне самому память нужна?
14)A: Используй библиотеку для своппинга (выгрузки из памяти) в XMS/EMS/Disk - перед запуском внешней программы твоя программа вместе со всеми данными переписывается в XMS/EMS или на винт, а после завершения внешней - восстанавливается. Можно воспользоваться, например, юнитом SPAWNO от Ральфа Брауна, или модулем OpExec из ObjectProf by TurboPower.

15)Q:> Как сделать процедуру с нефиксированным числом параметров, типа writeln?
15)A: Встроенными средствами BP - Никак. WriteLn - это не процедура вовсе, это макрос такой, сделанный для удобства.

16)Q:> Как с этим боpоться (сделать одну пpоцедуpу)?
16)A: Пеpедавать yказатель на блок паpаметpов и какой-нибyдь флаг, по котоpомy внyтpи пpоцедypы бyдешь сам pазбиpаться с этим блоком паpаметpов. Hапpимеp так: procedure aaa(var param; flag: byte);

17)Q:> Как узнать ErrorLevel,который выставила программа, запущенная через Exec()?
17)A: С помощью DosExitCode:


{$M 8192,0,0}
 Uses Dos;
 Begin
 SwapVectors;
 Exec(FExpand(FSearch('arj.exe',GetEnv('PATH'))),'x a.arj');
 SwapVectors;
 WriteLn('Код возврата=',lo(DosExitCode));
 End.
 

18)Q:> Предположим, запустили мой ЕХЕ. > Как узнать из программы, в каком каталоге он лежит, > и не переименовали ли его?
18)A: Функция ParamStr(0) возвращает полное имя запущеннонго exe-файла, например 'c:\exe\work.exe'. При запуске программы из-под IDE (Ctrl-F9) вместо имени программы ParamStr(0) вернет путь к файлу turbo.exe, если программа компилируется в память (Compile/Destination = Memory), или имя программы без пути, если компилировать на диск.

19)Q:> Как узнать, какие параметры передаются моему ЕХЕ при вызове из командной строки?
19)A: Функция ParamStr(i) возвращает i-й параметр в виде строки. Отдельным параметром считается комбинация символов, не содержащая пробелов. Функция ParamCount возвращает общее количество параметров, переданных программе. 19)A2: Если вам необходимо совершить некие продвинутые операции с переданной командной строкой - например, вы получаете параметры, содержащие пробелы и заключенные в кавычки, - вам поможет следующий исходник, копирующий всю командную строку в первоначальном виде в строковую переменную:


var
 s: ^string;
 begin
 s:=ptr(prefixseg, $80);
 writeln('Command line: "', s^, '"');
 end.
 

20)Q:> А можно записать в середину текстового файл строку произвольной длины, но чтобы остальные строки остались, как были, или удалить одну строку из середины?
20)A: Просто так - нет. Есть два варианта. Простой: открываете второй файл, читаете строки из первого, пишете строки во второй, пропуская то, что не нужно, или дополняя в нужных местах. Сложный: открываете файл, как бинарный, составляете список длин строк, и копируете куски файла в другой с помощью BlockRead/BlockWrite.


{пример для простого варианта}
 var
 f1,f2: text;
 s: string;
 const
 ourstring = 'ДЬВ'; {искомая стpока}
 begin
 assign(f1,'c:\a.txt');
 assign(f2,'c:\b.txt');
 reset(f1); rewrite(f2);
 while not eof(f1) do
 begin
 readln(f1,s);
 if pos(ourstring,s)<>0 then continue; {пропустить 
 строку, если найдено}
 writeln(f2,s);
 end;
 close(f1); close(f2);
 end.


Copyright MyCorp © 2025
Сделать бесплатный сайт с uCoz