Передача значений как массива

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

Sum( [1, 5, X, a+b*2] );


Передача массива вариантного типа

Массив, описанный как array[1..MaxNum] ofVariant; может быть передан в качестве параметра в подпрограмму, если этот параметр описан как array of const:procedureSum( A: array of const ),

Внутри тела подпрограммы надо анализировать перед обработкой тип каждого элемента этого массива.


Способы вызова подпрограмм

Подпрограммы с точки зрения прикладного программиста вызываются всегда оди-наково, но машинный код, который создается компилятором, для разных подпрограмм может отличаться. Это зависит от целей применения конкретной подпрограммы. Она может использоваться:
>> в рамках разрабатываемой прикладной программы;
>> как функция, доступная из динамической библиотеки .DLL;
>> как процедура, вызываемая из внешних программ или из Windows и т. д.
Для каждого из таких нестандартных случаев вслед за заголовком подпрограммы (за точкой с запятой) должно следовать одно из ключевых слов.

Таблица 1.10. Ключевые слова в заголовке подпрограмм
____________________________________________________________________

Ключевое слово Способ передачи параметров
_____________________________________________________________________
pascalСтандартный (параметры помещаются в стек)

registerСпособ, применяемый по умолчанию. Аналогичен использованию
ключевого слова pascaL но параметры передаются с помощью трех регистров
процессора, а не помещаются в стек (область оперативной памяти), что обычно приводит к повышению быстродействия программы

cdeclВ соответствии с соглашениями компиляторов для языков
программирования С и С++. Применяется, когда происходит обращение к
динамическим библиотекам DLL, написанным на этих языках

stdcallВ соответствии с соглашениями Windows

safecallИспользуется при работе с компонентными технологиями
_____________________________________________________________________

При использовании ключевых слов registerи pascal вычисление параметров выполняется слева направо и располагаются они в оперативной памяти перед вызовом подпрограммы в таком же порядке. При использовании ключевых слов cdecl, stdcallи safecallпараметры располагаются в обратном порядке (справа налево).

procedure Sum( A: array of const ); stdcall;


Существует еще одно зарезервированное слово Паскаля, forward, которое, при указании вслед за заголовком, говорит компилятору о том, что в данном месте расположен только заголовок подпрограммы, а все ее описание находится в исходном тексте далее. Такое описание обычно применяют, если в тексте имеется несколько подпрограмм, которые вызывают друг друга по кругу. Например, из процедуры Р1 вызывается процедура Р2, а из процедуры Р2 — процедура Р1.

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

Неправильно:
Подпрограммы:

procedurePI;
begin
P2;
end;

procedureP2;
begin
PI;
end;

Правильно:

procedureP2; forward;
procedurePI;
begin
P2;
end;

procedureP2 ;
begin
PI;
end;

Теперь компилятор знает, как выглядит заголовок процедуры Р2, и может корректно сгенерировать машинный код для обращения к ней.

Перегружаемые подпрограммы
Хотя в Паскале не допускается использование одинаковых названий для переменных, констант и других идентификаторов, для локальных переменных и подпрограмм делается исключение. Так как в Паскале предъявляются строгие требования к типам данных, обращаться к подпрограмме, формальные параметры которой имеют тип Integer, с фактическими параметрами, имеющими тип Real, нельзя. Однако при решении задачи подчас бывает необходимо, чтобы подпрограмма с одним и тем же именем работала с разными типами данных.
Здесь есть два способа действий: либо использовать данные типа Variant (что чревато ошибками преобразования, снижает общую эффективность программы и требует от разработчика повышенной бдительности), либо применить перегружаемые подпрограммы. Они отличаются от обычных подпрограмм тем, что имеют совпадающие имена, а различаются только типами аргументов. Чтобы указать компилятору, что конкретная подпрограмма — перегружаемая, надо вслед за ее заголовком указать зарезервированное слово overload.

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


procedureOvl( X: Real ); overload;

begin
. . .
end ;

procedureOvl( X: Byte ); overload;

begin
. . .
end;

Ovl( 1 ); // вызывается процедура Ovl( X: Byte )

Ovl( 1.0 ); // вызывается процедура Ovl( X: Real )


Необходимое требование к перегружаемым процедурам состоит в том, чтобы списки параметров совпадали во всем, за исключением типов переменных.
Надо особенно осторожно использовать перегружаемые подпрограммы с параметрами по умолчанию. Например:

 

procedureOvl( X: Byte; Y: Real = 1 ); overload;
begin
. . .
end;

procedureOvl( X: Byte ); overload;
begin
. . .
end;

При вызове

Ovl(1)

 

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


Локальное описание

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


Вложенные подпрограммы

Помимо обычных описаний, внутри подпрограммы допускается объявлять также локальные подпрограммы, к которым можно обращаться (вызывать) только из тела «родительской» подпрограммы. При этом локальная подпрограмма может свободно обращаться к любым локальным описаниям (переменным, типам), которые расположены до описания данной подпрограммы.
Такая возможность полезна, когда во время кодирования подпрограмма начинает непредвиденно разрастаться. Ее приходится делить на более мелкие фрагменты, которые в то же время желательно не выносить за пределы текущей подпрограммы, чтобы иметь возможность пользоваться ранее сделанными локальными описаниями. Например:


procedureDemo;
type Tl = array[1..2] of Real;
varD, Dl: Tl;
S: Real;

procedureInDemo;
begin
Dl := D;
S := Dl[l] + Dl[2]
end;
begin
. . .
end;

 


Уровень вложенности локальных подпрограмм неограничен. Тело








Дата добавления: 2014-12-01; просмотров: 1449;


Поиск по сайту:

При помощи поиска вы сможете найти нужную вам информацию.

Поделитесь с друзьями:

Если вам перенёс пользу информационный материал, или помог в учебе – поделитесь этим сайтом с друзьями и знакомыми.
helpiks.org - Хелпикс.Орг - 2014-2024 год. Материал сайта представляется для ознакомительного и учебного использования. | Поддержка
Генерация страницы за: 0.01 сек.