Студопедия КАТЕГОРИИ: АвтоАвтоматизацияАрхитектураАстрономияАудитБиологияБухгалтерияВоенное делоГенетикаГеографияГеологияГосударствоДомЖурналистика и СМИИзобретательствоИностранные языкиИнформатикаИскусствоИсторияКомпьютерыКулинарияКультураЛексикологияЛитератураЛогикаМаркетингМатематикаМашиностроениеМедицинаМенеджментМеталлы и СваркаМеханикаМузыкаНаселениеОбразованиеОхрана безопасности жизниОхрана ТрудаПедагогикаПолитикаПравоПриборостроениеПрограммированиеПроизводствоПромышленностьПсихологияРадиоРегилияСвязьСоциологияСпортСтандартизацияСтроительствоТехнологииТорговляТуризмФизикаФизиологияФилософияФинансыХимияХозяйствоЦеннообразованиеЧерчениеЭкологияЭконометрикаЭкономикаЭлектроникаЮриспунденкция |
ГРАММАТИКА ЯЗЫКОВ ПРОГРАММИРОВАНИЯ
Описанию грамматики языка предшествует описание его алфавита. Алфавит любого языка состоит из фиксированного набора символов, однозначно трактуемых. Алфавит языков программирования, как правило, связан с литерами клавиатуры печатной машинки. Клавиатуры персональных компьютеров близки к ним по наличию литер. Алфавиты большинства языков программирования близки друг другу и основываются на буквах латинского алфавита, арабских цифрах и общепринятых спецсимволах, таких как знаки препинания, математических операций, сравнений и обозначений. Большинство популярных языков программирования в своем алфавите содержат следующие элементы:
<буква> : : = AaBbCcDdEeFf и т.д. <цифра> ::=0123456789 <знак арифметической операции >::=*/+- <разделитель> ::=.,;:()[]{}':= <служебное слово> :: = begin end if then else for next и т.д. <спецсимвол> :: = <знак арифметической операции> | <разделитель> | <служебное слово> <основной символ>::=<буква> | <цифра> | <спецсимвол> <комментарий>::=<любая последовательность символов>
Несмотря на значительные различия между языками программирования, ряд фундаментальных понятий в большинстве из них схожи. Приведем часть этих понятии. Оператор - одно из ведущих понятий всех языков программирования (теоретически, за исключением чисто декларативных; но в действительности и они используют родственное понятие). Каждый оператор представляет собой законченную фразу языка и определяет однозначно трактуемый этап обработки данных В соответствии с теорией алгоритмов выделяют основные (базисные) операторы языка: присвоения, условный и безусловный переход, пустой оператор. К производным, не основным, относят составной оператор, оператор выбора, оператор цикла и оператор присоединения. Все операторы языка в тексте программы отделяются друг от друга явными или неявными разделителями, например:
Sl;S2;...;Sn Операторы выполняются в порядке их следования в тексте программы. Лишь с помощью операторов перехода этот естественный порядок может быть нарушен. Большая часть операторов ведет обработку величин.Величины могут бытьпостоянными и переменными. Значения постоянных величин не изменяются в ходе выполнения программы. Величина характеризуетсятипом, именем и значением.Наиболее распространенные типы величин - числовые (целые и вещественные), символьные, логические. Тип величины определяется ее значением. Другая важная классификация величин - простые и структурированные. Простая величина в каждый момент может иметь не более одного значения. Ей соответствует одна ячейка памяти (поскольку термин «ячейка» несколько устарел, часто говорят «машинное слово») или ее эквивалент во внешней памяти компьютера. Структурированная величина, имея одно имя, может иметь разом несколько значений. Эти значения представляют собой элементы (компоненты) величины. Самый широкоизвестный пример - массив, у которого элементы различаются по индексам (номерам). Вопрос о структурировании величин - входных, выходных и промежуточных - для успеха решения прикладной задачи не менее важен, чем вопрос о правильном написании последовательности операторов. Важнейшие характеристики структурированной величины таковы: упорядоченность (да или нет), однородность (да или нет), способ доступа к элементам, фиксированность числа элементов (да или нет). Так, массив является упорядоченной однородной структурой с прямым доступом к элементам и фиксированным их количеством. Всем программным объектам в языках даются индивидуальныеимена. Имя программного объекта называют идентификатором (от слова «идентифицировать»). Чаще всего идентификатором является любая конечная последовательность букв к цифр, начинающаяся с буквы:
<идентификатор>::=<буква> | <идентификатор> | <буква> <идентификатор><цифра>
Как правило, в большинстве языков программированияв качестве идентификатора запрещается использовать служебные слова языка. Многим слово «идентификатор» не нравится, и в настоящее время чаще употребляют слово «имя», поскольку
<имя>::=<идентификатор>.
Программисты выбирают имена по своему усмотрению. Принципы выбора и назначения имен программным объектам естественны. Следует избегать мало выразительных обозначений, не гоняться за краткими именами. Имена должны быть понятны, наглядны, отражать суть обозначаемого объекта. Например,
Summa, Time, i, j, integral, init и т. п.
Некоторым идентификаторам заранее предписан определенный смысл и их называют стантартными, например, Sin - это имя известной математической функции. Описания или объявления программных объектов связаны с правилами обработки данных. Данные бывают разные и необходимо для каждого из них определить его свойства. Например, если в качестве данных выступает массив, то необходимо задать его размерность, границы индексов, тип элементов массива. Описательная часть языка программирования является необходимой как для системных программистов - разработчиков трансляторов, которые должны, в частности, проводить синтаксическую и семантическую диагностику программ, - так и для «прикладного» программиста, которому объявления программныхобъектов частооблегчают процесс разработки и отладки программ. В некоторых языках стандартные описания простых числовых и символьных данных опускают (описания по умолчанию), или в них задаются правила описания по имени объекта. Например, в Фортране переменные, имена которых начинаются с букв I, J, К, L, M, N, могут принимать целые значения (при отсутствии явного описания типа, которое возможно), т.е. определены как числовые данные целого типа. В Бейсике-MSX данные строкового типа присваиваются переменным, имена которых заканчиваются специальным символом $: A$, S1$. Особый интерес представляют в языках программирования описания нестандартных структур данных, таких как запись, файл, объект, список, дерево и т.п. Приведем список наиболее употребительных обозначений типов данных, используемых в описаниях:
Целый - Integer Вещественный - Real Логический - Boolean Символьный - Char Строковый - String Массив - Array Множество -Set Файл - File Запись - Record Объект - Object Переменные играют важнейшую роль в системах программирования. Понятие «переменная» в языках программирования отличается от общепринятого в математике. Переменная - это программный объект, способный принимать некоторое значение с помощью оператора присваивания. В ходе выполнения программы значения переменной могут неоднократно изменяться. Каждая переменная после ее описания отождествляется с некоторой ячейкой памяти, содержимое которой является ее значением. Синтаксис переменной, точнее, ее идентификатора, как правило, имеет вид:
<имя переменной>::= ——><буква>———> —><буква>———> —><цифра>——> —><спецсимвол>
Семантический смысл переменной заключается в хранении некоторого значения. соответствующего ее типу (например, переменная целого типа может принимать значение произвольного целого числа), а также в выполнениис ней операций пересылки в нее и извлечения из нее этого значения. Функция - это программный объект, задающий вычислительную процедуру определения значения, зависимого от некоторых аргументов. Вводится в языки программирования для задания программистом необходимых ему функциональных зависимостей. В каждом языке высокого уровня имеется в наличии библиотека стандартных функций: арифметических, логических, символьных, файловых и т.п. Функции -стандартные и задаваемые программистом - используются в программе в выражениях. Выражения строятся из величин - постоянных и переменных, функций, скобок. знаков операций и т.д. Выражение имеет определенный тип, определяемый типом принимаемых в итоге его вычисления значений. Возможны выражения арифметические, принимающие числовые значения, логические, символьные, строковые и т.д. Выражение 5+7 является, несомненно, арифметическим, выражение А + В может иметь cамый разный смысл - в зависимости от того, что стоит за идентификаторами А и В. Процедура - это программный объект, представляющий некоторый самостоятельный этап обработки данных. По сути, процедуры явились преемниками подпрограмм, которые были введены для облегчения разработки программ еще на самых ранних стадиях формирования алгоритмических языков. Процедура имеет входные и выходные параметры, называемые формальными. При использовании процедуры формальные параметры заменяются на фактические. Модуль (Unit) -это специальная программная единица, предназначенная для создания библиотек и разделения больших программ на логически связанные блоки. По сути, модуль - это набор констант, типов данных, переменных, процедур и функций. В состав модуля входят разделы: заголовок, интерфейс, реализация, инициализация. Заголовок необходим для ссылок на модуль. Интерфейс содержит объявления, включая процедуры и функции. Раздел «реализация» содержит тела процедур и функций, перечисленных в интерфейсной части. Раздел «инициализация»содержит операторы, необходимые для инициализации модуля. Каждый модуль компилируется отдельно, и каждый элемент модуля можно использовать в программе без дополнительного объявления.
Контрольные вопросы и задания
1. Какие преимущества имеют языки программирования высокого уровняпосравнению с машинно-ориентированными языками? 2. Каковы основные составляющие языка программирования высокого уровня? 3. В чем различия понятий языков программирования от аналогичных понятий математического «языка»? 4. С какой целью используются и что представляют собой металингвистические формулы Бэкуса-Наура? 5. Что представляет собой синтаксическая диаграмма Вирта? 6. В чем различие между постоянными и переменными величинами? Чем характеризуется величина? 7. В чем принципиальная разница между величинами простыми и структурированными? 8. Для чего служит описание величин в программах? 9. В чем состоит назначение функций? процедур? модулей?
§3. ПАСКАЛЬ КАК ЯЗЫК
ВВЕДЕНИЕ
Язык Паскаль, начиная с момента своего создания Н.Виртом в 1971 г., играет особую роль и в практическом программировании, и в его изучении. С непревзойденной четкостью в нем реализованы принципы структурного программирования, которые мы рассматривали в п. 1.8. Паскаль стал первым языком, с которым знакомится большинство будущих программистов в мире. Трансляторы для программ, написанных на Паскале, разработаны для различных компьютеров и в настоящее время имеют множество разновидностей. Они являются компиляторами, обрабатывающими разработанные программистами тексты программ. Существует много версий языка Паскаль. Различия между ними порой весьма велики. Так, базовая версия Вирта имеет многократно меньшие возможности чем, скажем, версия Турбо-Паскаль 7.0 (первая, фактически - язык для обучения будущих программистов, а вторая - орудие профессиональных разработчиков прикладного программного обеспечения). Тем не менее, это версии одного языка, что, в частности, подтверждается их совместимостью «сверху вниз», т.е. любая программа, соответствующая «младшей» версии, соответствует и «старшей» (за исключением малозначащих синтаксических оговорок). Приведенные ниже тексты программ и примеры соответствуют (если нет специальных оговорок) версиям не ниже Турбо-Паскаль 3.0. Любая Паскаль-программа является текстовым файлом с собственным именем и с расширением .pas. Рассмотрим в качестве примера текст программы 1 решения квадратного уравнения. Паскаль-программа имеет вид последовательности символов латинских и русских букв, арабских цифр, знаков операции, скобок, знаков препинания и некоторых дополнительных символов. В ней можно выделить описания данных и операторы, описывающие действия, которые надо выполнить машине над этими данными. Программа I program KvadUravn; (заголовок программы) var (список переменных) а,b,с: real; (коэффициенты уравнения) d,xlx2: real; (вспомогательные переменные) begin (начало программы) writein; (пропуск строки на экране) writein('введи a,b,c'); read(а,Ь,с); (ввод данных) d:=b*b-4*a*c; (дискриминант) if d<0 then (если d<0, то) write('корней нет') (печатать) else (иначе) begin (начало серии команд) xl:=(-b+sqrt(d))/(2*a); x2:=(-b-sqrt(d))/(2*а); (вычисляем корни) write('х1=',х1,' х2=',х2) (печать корней) end (конец серии) end. (конец программы)
Схематически программа представляется в виде последовательности восьми разделов: 1) заголовок программы; 2) описание внешних модулей, процедур и функций; 3)описание меток; 4) описание констант; 5) описание типов переменных; 6) описание переменных; 7) описание функций и процедур; 8) раздел операторов. Не в каждой программе обязательно присутствуют все восемь разделов, в простейшей программе, например, могут быть только 5-й и 8-й разделы. Каждый раздел начинается со служебного слова, назначение которого зафиксировано в Паскале так, что его нельзя употреблять для других целей (список и перевод служебных слов дан в Приложении). Рассмотрим пример программы 2, вычисляющей длину окружности и площадь круга по данному радиусу. Программа 2 program circle; const pi=3.14159; var r,s,1 : real; begin writeln (введите радиус'); readln(r); s:=pi*r*r; l:=2*pi*r; writeln('площадь круга=',S:8:4); writeln('длина окружности=', l:8:4) end.
В этой программе можно выделить четыре раздела. Описание заголовка начинается со служебного слова program, описание констант - const, описание переменных -var, раздел операторов начинается с begin. Программа заканчивается служебным словом end, после которого ставится точка. Описания величин и операторы друг от друга отделяются знаком «точка с запятой». Для обозначения величин используются имена. Они составляются из латинских букв и цифр, причем первым символом должна быть буква. В примере использованы имена величин - pi, r,s и 1. Имя программы (в примере - circle) выбирается автором и составляется по такому же правилу. Постоянные величины (константы) чаще всего бывают числовыми или символьными (но могут быть и других типов, о которых речь пойдет ниже). Значения символьных констант заключаются в апострофы. Постоянные величины описываются в разделе констант по схеме:
const <имя> = <константа>
В разделе констант может быть описано несколько постоянных величин. Например:
const pi=3.14159; k=-15; s='площадь';
Данные, обрабатываемые программой, могут быть разных типов (числа, символы, строки, массивы и т.д.). Тип определяет область допустимых значений, а также операции и функции, применяемые к величинам этого типа. В Паскале имеется несколько встроенных простых типов со стандартными именами. Группа типов, значения каждого из которых можно перечислить в некотором списке- скалярные типы. Для них определена порядковая функция ord(x) - номер значения х в списке (для целочисленного х ord(x)=x); функции pred(x) - значение в списке, предшествующее х, и suce(x) - значение в списке, следующее за х. Упорядоченный тип - это тип, значения которого упорядочены в обычном смысле. К данным такого типа применимы операции отношения <,>, <= (меньше или равно), >=(больше или равно), =, <> ( не равно). Для логического типа выполняется неравенство:
false < true.
Переменные описываются в разделе описания переменных посхеме:
var <список имен переменных>: <тип>
Имена в списке разделяются запятой. В этом разделеможет быть описано несколько переменных разного типа, например:
var a,b,c:real; k,l:integer; p:boolean;
Надцелыми величинами (тип integer) определены арифметические операции : * (умножение), div (деление нацело), mod (вычисление остатка от деления), + , -(сложение и вычитание); операции перечислены в порядке старшинства. Например: 25 div 4 = 6 ; 25 mod 4=1. Целый результат дают некоторые стандартные функции (аргумент функции заключается в круглые скобки):
abs(x) - абсолютная величина целого х; sqr(x) -квадрат значения х; trunc(x) - целая часть вещественной величины х; round(x) - целое число, полученное из вещественного х по правилу округления; random(x) - случайное целое числоиз интервала от 0 до х.
Например: trunc(4.7)=4 ; round(4.7)=5 ; sqr(3)=9 . Для данных типа byte определены те же операции и функции, что и для данных типа integer. Над вещественными величинами определены операции: *, +, -, /, а также стандартные функции при вещественном или целом аргументе: abs(x), sqr(x), sin(x), cos(x), arctan(x), ln(x), exp(x), sqrt(x) - квадратный корень из х, int(x) - целая частьизх, random - случайное число из интервала от 0 до 1. Указанные операции и функции дают вещественный результат. Надлогическими величинами определены операции: not - отрицание, and -конъюнкция, or-дизъюнкция. Логическая функция odd(x) принимает значение true, если целочисленное х является нечетным и false , если четным. Множество всех символов образуютсимвольные величины (тип char), которые являются упорядоченными, причем 'А' < 'В'< 'С'<...< 'Z', 'а' < 'b'<...< 'z', '0' < 'I'<...< '9'. Выражения - это конструкции, задающие правила вычисления значений переменных. В общем случае выражения строятся из переменных, констант, функций с помощью операций и скобок. Эта роль выражений отражена в основном операторе языка - операторе присваивания. Он имеет следующий вид: <имя переменной> := <выраженне> Тип переменной и тип выражения должны быть согласованы (величины принадлежат одному и тому же типу). Есть исключение: имя переменной может относиться к типу real, а значение выражения - к типу integer. Примеры. l:=2*pi*r; p:=(a+b+c)/2; z:=sqrt(sqr(x)+sqr(y)) В Паскале можно вводить с клавиатуры числовые и символьные данные. Имеются две встроенные процедуры (подпрограммы) ввода: 1) read(<cписок переменных>); 2) readln(<cписок переменных>). При выполнении процедуры read(xl,x2,...,xN) программа прерывается и компьютер ждет ввода с клавиатуры N значений переменных из списка х1, х2,..., xN. Эти значения - константы соответствующих типов - должны при вводе разделяться пробелами. Набор данных завершается клавишей ввода. Процедура readin отличается от read только тем, что при завершении ввода курсор перемещается в начало следующей строки. Пример.
var a,b:real; c:char; d:integer; ... read(a,c,d,b); ...
Допустимый ввод: 83.14 k 200-7.15 Программа на Паскале может выводить на экран или на принтер значения числовых или символьных выражений. Имеются две процедуры вывода на экран: 1) write(<cписок выражений>); 2) writeln(<cписок выражений>). Процедура write(xl,x2,...,xN) печатает на экране значения выражений из списка х1, х2, ..., xN. Процедура writeln отличается от write тем, что переводит курсор в начало следующей строки. Для вывода на принтер используются те же процедуры с добавлением служебного слова 1st перед списком выражений. Пример: write(lst,'нет решений'); На бумаге будет напечатан текст «нет решений». Для управления печатью используютсяформаты данных. Пусть х - переменная типа real. Если не использовать форматы, то значение х будет выводиться в «плавающей» форме (типа 1.654887892Е-04). Форматы позволяют напечатать вещественное число в естественной форме. Пусть m, n - целые числа. Процедура write(x:m:n) выводит на экран значение переменной х в виде десятичной дроби, причем m определяет общее число выводимых символов, включая цифры, точку и знак числа, n - количество цифр после точки. Если количество выводимых символов меньше m, то перед числом добавляются пробелы. Пусть, например, х = 387.26. Следующая таблица демонстрирует влияние форматов на вывод значения х: оператор строка вывода writeln('*',x) * 3.8726000000E+02 writeln('*',x:8:3) * 387.260 writeln('*',x:8:l) * 387.3 Один формат - ширину поля вывода - можно использовать и для вывода значений выражений типов integer, boolean, char. Контрольные вопросы 1. Какова последовательность разделов в программе на Паскале? 2. Какие типы данных называются скалярными? упорядоченными? 3. Какие действия допустимы над величинами целого типа? вещественного типа? 4. Как в Паскале осуществляется ввод и вывод данных? ОСНОВНЫЕ КОНСТРУКЦИИ ЯЗЫКА Паскаль - язык структурного программирования. Это означает, что программист должен выражать свои мысли очень дисциплинированно, с использованием малого числа четко оговоренных конструкций, используя как чередование их, так и вложения друг в друга. Не рекомендуется (хотя и возможно) использовать оператор перехода goto. Реализация последовательности действий (т.е. структуры следования) выполняется с помощью составного оператора: begin <последовательность операторов> end Раздел операторов в программе всегда является составным оператором. Служебные слова begin и end часто называют операторными скобками. Для реализацииразвилки в Паскале предусмотрены два оператора: условный оператор и оператор варианта (выбора). Они предназначены для выделенияизсоставляющих их операторов одного, который и выполняется. Структура и действиеусловного оператора таковы: If <логическое выраженне> then <оператор 1> else <оператор 2> Условный оператор может быть неполным,т.е. не содержать часть «else <оператор 2>». В этом случае, если значение логического выражения равно false, условный оператор не вызывает никаких действий. Пример: составим программу, которая определяет длину общей части двух отрезков числовой оси, заданных координатами своих концов соответственно а, b и с, d (а < b, с < d). Если отрезки имеют общую часть, то левая координата общей части отрезков m равнамаксимальному из чисел а и с, а правая n - минимальному из чисел b и d. Программа 3
program cross; var a,b,c,d,m,n,l:real; begin writeln('введите координаты концов отрезков') ; read(a,b,с,d) ; writeln; if a<c then m:=c else m:=a; if b<d then n:=b else n:=d; if m<n then l:=n-m else 1:=0; writeln("длина общей части отрезков=',1:6:2) end. Оператор варианта имеет следующую форму: case <выражение> of <список констант 1> : <оператор 1>; <список констант 2> : <оператор 2>; ………………………… <список констант N> : <оператор N> end.
Выражение, стоящее между служебными словами case и of, должно иметь значение ординального типа. Любой список констант может состоять из одной константы. Оператор варианта вычисляет значение выражения, записанного после case. Если его значение совпадает с одной из констант в некотором списке, то выполняется оператор, стоящий после этого списка. Если значение выражения не совпало ни с одной константой во всех вариантах, то оператор варианта ничего не делает. В качестве примера приведем программу, которая в зависимости от номера месяца выдает сообщение о времени года. Программа 4 program seasons; var k:integers begin writeln('введите номер месяца') ; readin(k); case k of 1,2,12:writeln('зима'); 3, 4,5:writeln('весна') ; 6, 7,8:writeln('лето'); 9,10,11:writeln('осень') end end.
Для реализации циклов в Паскале имеются три оператора Если число повторений известно заранее, то удобно воспользоваться оператором цикла с параметром. В других случаях следует использовать операторы цикла с предусловием (цикл «пока» ) или с постусловием (цикл «до»). Цикл с предусловием является наиболее мощным в Паскале. Другие операторы цикла можно выразить через него. Его форма такова:
while - <логическое выражение> do <оператор>
Действие: вычисляется значение логического выражения. Если оно равно true, то выполняется оператор, после чего снова вычисляется значение логического выражения, в противном случае действие заканчивается. В качестве примера использования такого цикла решим следующую задачу. На склад привозят однородный груз на машинах различной грузоподъемности. На компьютер, управляющий работой склада, поступает информация о весе груза очередной машины. Составить программу подсчета количества машин,прибывшихна склад до его заполнения, если вместимость склада не более 100 тонн. Введем обозначения: sum - сумма веса груза, хранящегося в этот момент на складе; num - число разгруженных машин; w - масса груза очередной машины. Вначале величины sum и num равны нулю. Цикл разгрузки продолжается, пока выполняется неравенство sum < 100.
Программа 5 program store; var sum,w:real; num:integer; begin num:=0;sum:=0; while sum<100do begin writeln('введите вес груза машины'); readln(w); sum:=sum+w; if sum<=100 then num:=num+l else writeln('груз уже не поместится') end; writeln('число разгруженных машин =',num;3) end.
Операторцикла с постусловием имеет форму: repeat <последовательность операторов> until <логическое выражение> Действие: выполняется последовательность операторов. Далее вычисляется значение логического выражения. Если оно равно true, то действие заканчивается, в противном случае снова выполняется последовательность операторов и т.д. Решим предыдущую задачу, применяя цикл с постусловием. Цикл разгрузки заканчивается, если выполняется условие: sum > 100. Программа 6 program store2; var sum,w:real; man:integer; begin num: =0; sum: =0; repeat writeln('введите вес груза машины'); readln(w); sum:=sum+w; if sum<=100 then num:=num+l . else writeln('груз ухе не поместится') until sum>=100; writeln('количество разгруженных машин =',num:3) end. Операторцикла с параметром предусматривает повторное выполнение некоторого оператора с одновременным изменением по правилу арифметической прогрессии значения управляющей переменной (параметра) этого цикла. Оператор цикла с параметром имеет две формы. Форма1: for <параметр>:= <выражение1> to <выражение 2> do <оператор> Параметр, выражение 1, выражение 2 должны быть одного ординального типа; Параметр в этом цикле возрастает. Действие эквивалентно действию следующего составного оператора: begin <параметр>:=<выражение 1>; while <параметр> <= <выражение 2> do begin <оператор>; <параметр>:=suсс(<параметр>) end end. Если в этом описании отношение <= заменить на >= , а функцию succ на pred, то параметр в цикле будет убывать, в этом случае цикл с параметром принимает форму 2. Форма 2: for <параметр>:=<выражение 1> downto <выражение 2> do <оператор> Пример: составим программу, по которой будет напечатана таблица перевода километров в мили (1 миля = 1,603 км). Параметром цикла можно считать целую величину k - количество километров. Пусть эта величина изменяется от 1 до 10 (с шагом 1, разумеется). Программа 7 program mili; const a=1.603;b='км'; с='мили'; var k:integer; m:real; begin writeln(b:5,c:7); writeln; for k:=l to 10 do begin m:=k/a; writeln(k:5,m:6:3) end end. Запишем в этой программе цикл с параметром в форме 2: for k:=10 downto 1 do begin m:=k/a; writeln(k:5,m:6:3) end.
Тогда значения k в таблице будут убывать от 10 до 1 с шагом 1. Контрольные вопросы
1. Как в Паскале реализуется развилка? 2. В чем различие в назначениях условногооператора и оператора варианта? Как реализуется оператор варианта? 3. Какого типа циклы реализуются в языке Паскаль? каким образом? СТРУКТУРЫ ДАННЫХ
Мы уже познакомились с простыми типами real, integer, boolean, byte,char. В Паскале программист по своему желанию может определить новый тип путем перечисления его элементов - перечисляемый тип, который относится к простым ординальным типам. Описание перечисляемого типа выполняется в разделетипов по схеме: type <имя типа> = <список имен>
Примеры:
type operator = (plus,minus,multi, divide); color = (white,red,blue,yelow,purple,green);
В списке должно быть не более 256 имен. Поскольку перечисляемый тип относится к ординальным, то к его элементам можно применять функции ord(x), pred(x), succ(x) и операции отношения. Отметим, что данные этого типа не подлежат вводу и выводу с помощью функций ввода/вывода и могут использоваться внутри программы для повышения ее читабельности и понятности. Для иллюстрации работы с перечисляемыми типами приведем программу 8, переводящую английские названия дней недели на русский язык.
Программа 8 programweek; type days=(mon,tue,wed,thu,fri,sat,sun); var d:days; begin for d:=mon to sun do case d of mon: writeln("понедельник"); tue writeln("вторник"); wed writeln("среда"); thu writeln("четверг"); fri writeln("пятница"); sat writein("суббота") ; sun writeln("воскресенье") end end.
Интервальный тип - это подмножество другого уже определенного ординального типа, называемого базовым. Интервал можно задать в разделе типов указанием наименьшего и наибольшего значений, входящих в него и разделяющихся двумя последовательными точками, например:
type days=(mon,tue,wed,thu,fri,sat,sun); workday s=mon.. fri; index=1..30; letter='a'..'z';
Можно задать интервал и в разделе переменных:
vara:1..100;b:-25..25;
Операции и функции - те же, что и для базового типа. Использование интервальных типов в программе позволяет экономить память и проводить во время выполнения программы контроль присваиваний. Пример: если k - номер месяца в году, то вместо описания
var k:integer;
можно написать
vark:1..12;
Интервальный тип тоже относится к простым ординальным типам.
СОСТАВНЫЕ СТРУКТУРЫ. Данные, с которыми имеет дело ЭВМ, являются абстракциями некоторых реальных или существующих в воображении людей объектов. Мы уже познакомились с некоторыми типами данных: стандартными, перечислимыми, интервалами. Этими типами можно было обойтись при решении простых задач. Однако естественно и часто очень удобно группировать однотипные данные в последовательности - массивы, строки символов, объединять разнотипные данные об одном и том же объекте в виде записей. Значительные удобства представляются пользователю в Паскале при организации однотипных величин в виде множества с соответствующим набором операций: объединения, пересечения и т.д. Наконец, последовательность однотипных величин переменной длины можно представить в Паскале в виде файла данных и хранить на внешних носителях, используя его в разных программах. Таким образом, подобно составному оператору, содержащему несколько других операторов, составные типы образуются из других типов, при этом существенную роль играет метод образования или структура составного типа. Часто используемый составной тип - массив. Массив - это последовательность, состоящая из фиксированного числа однотипных элементов. Все элементы массива имеют общее имя (имя массива) и различаются индексами. Индексы можно вычислять, их тип должен быть ординальным. При описании массивов используются служебные слова array и of. В описании массива указывается тип его элементов и типы их индексов. Схема описания такова:
type <имя типа> = array [<список типов индексов>] оf <тип элементов>
Тип элементов - произвольный, он может быть составным. Число типов индексов называется размерностью массива. После описания типа массива конкретные массивы можно задать в разделе описания переменных. Например:
type vector = array [1.. 10] of real; table = array ['A'..'Z',1..5] of integer; var a,b : vector; с: table;
Обращение к элементу массива осуществляется с помощью задания имени переменной, за которым следует заключенный в квадратные скобки список индексов элемента. Например:
а[7]:=3.1; b[k*k+l]:=0; с['М',3]:=-14;
Если массивы имеют одно и то же описание, то во многих версиях Паскаля допустимо их копирование, например b:=а; Описание массива можно совместить с описанием соответствующих переменных:
var a,b : array [1.. 10] of real; d : array [byte] of char;
В Турбо-Паскале разрешена инициализация начальных значений составных переменных с помощью, так называемых, типизированных констант. Типизированные константы используются как переменные того же типа. Их применение экономит память, однако они не могут быть использованы для определения других переменных того же типа. Схема описания констант массива:
const <имя массива>: <тип массива> = (<список значений элементов>)
Тип массива может быть описан ранее:
type digits = array [1 ..5] of char; const a : digits =('0';2','4';6';8');
Пример: используя массив, составим программу, которая напечатает на экране 20 чисел Фибоначчи. Последовательность Фибоначчи определяется равенствами
а[1]=а[2]=1; a[k]=a[k-l]+a[k-2] прик>2.
Использование массива позволяет создать эффективную программу. Для вывода каждого члена последовательности отведем на экране 5 позиций.
Программа 9 program fibon; "'' const n=20; var a: array[l..n] of integer; k: integer; begin a[l]:=l;a[2]:=l; for k:=3 to n do a[kl:=a[k-l]+a[k-2]; for k:=l to n do write(a[k]:5); writeln end.
Рассмотрим часто встречающуюся задачу упорядочения членов числовой последовательности по какому-либо признаку. Пример: упорядочить члены числовой последовательности по возрастанию. Используем метод упорядочения, носящий имя «пузырек». Будем просматривать пары соседних элементов последовательно справа налево и переставлять элементы в паре, если они стоят неправильно:
5,3,2,4,1 → 5,3,2,1,4 → 5,3,1,2,4 →5,1, 3,2,4 → 1,5,3,2,4
В начале просмотра присвоим некоторой логической переменной значение true: p:=true; если при просмотре пар была хотя бы одна перестановка, изменим значение логической переменной на противоположное: p:=false; это означает, что последовательность еще не была упорядочена и просмотр пар надо повторить. Цикл просмотров заканчивается, если после очередного просмотра выполняется условие: p=true. Последовательность зададим в программе как константмассивиз 10 элементов - целых чисел.
Программа 10 program bubble; ' const a:array[l..10] of integer=(19,8,17,6,15,4,13,2,11,0); var b,i:integer; p :boolean; begin c1rscr; for i:=l to 10 do write(a[i]:3);writeln;writeln; repeat p:=true; for i:=10 downto 2 do if a[i]<a[i-l] then begin
b:=a[il;a[i]:=a[i-l]; ali-1]:=b;p:=false end until p=true; for i:=l to 10 do write(a[i]:3); writeln end.
Обработка элементов двумерных массивов (матриц) обычно выполняется с помощью двойного цикла. Один цикл управляет номером строки, другой - номером столбца. При решении задач на ЭВМ часто возникает необходимость в использовании последовательностей символов. Такую последовательность можно описать как массив символов, однако в Паскале для таких целей имеется специальный тип - string[n] -строка из n символов, где n <= 255. Способы описания переменных - строк - аналогичны описанию массивов. 1. Строковый тип определяется в разделе описания типов, переменные этого типа - в разделе описания переменных:
type word : string[20]; var a,b,c : word;
2. Можно совместить описание строкового типа и соответствующих переменных в разделе описания переменных:
var a,b,c : string[20]; d : string[30];
3. Можно определить строковую переменную и ее начальное значение какконстант-строку: const l:string[l 1]='информатика';
Символы, составляющие строку, занумерованы слева направо; к ним можно обращаться с помощью индексов, как к элементам одномерного массива. Для переменных одного строкового типа определен лексикографический порядок, являющийся следствием упорядоченности символьного типа:
'fife' < 'tree' (таккак 'f'< 't'); '4' > '237' (так как '4' > '2').
Кроме логических операций<, >, =, для величин строкового типа определена некоммутативная операция соединения, обозначаемая знаком плюс:
а:='кол'+'о'+'кол'; (в результате а='колокол').
Для строковых величин определены следующие четыре стандартные функции. 1. Функция соединения - concat(sl,s2,...,sk). Значение функции - результат соединения строк sl ,s2,...sk, если он содержит не более 255 символов. 2. Функция выделения - copy(s,i,k). Из строки s выделяется k символов, начиная с i-того символа:
а:=сору('крокодил',4,3); (в результате а='код*).
3. Функция определения длины строки - length(s). Вычисляется количество символов, составляющих текущее значение строки s:
b:=length('каникулы'); (b=8).
4. Функция определения позиции - pos(s,t). Вычисляется номер позиции, начиная с которого строка s входит первый раз в строку t; результат равен 0, если строка s не входит в t:
с:=роs('ом','компьютер'); (с=2).
В Паскале определены также четыре стандартные процедуры для обработки строковых величин: 1. Процедура удаления delete(s,i,k).Из строки s удаляется k символов, начиная с i-того символа.
s:='таракан'; delete(s,5,2); (в результате S='таран').
2. Процедура вставки - insert(s,t,i). Строка s вставляется в строку t, начиная с позиции i:
t:='таран'; insert ('ka',t,5); (t='таракан').
3. Процедура преобразования числа в строку символов - str(k,s). Строка s получается «навешиванием» апострофов на число k:
str(564,s); (s='564').
4. Процедура преобразования строки из цифр в число - val(s,k,i). Число i=0, если в строке s нет символов, отличных от цифр, в противном случае i=позиции первого символа, отличного от цифры:
val('780',k,i); (k=780; i=0).
Рассмотрим несколько программ, в которых используются строковые величины. 1. Составить программу, определяющую количество гласных в русском тексте, содержащем не более 100 символов. Здесь удобно определить констант-строку, состоящуюиз всех 18 строчных и заглавных русских букв, и в цикле проверить, будет ли очередной символ заданного текста элементом констант-строки.
Программа 11 program vowel; const с:зtring[18]='аеиоуыэюяАЕИОУЫЭЮЯ' ; var a :string[100]; k,n:integer; begin writeln('введите текст'); readln(a);n:=0; for k:=l to length(a) do if pos(a[k],c)>0 then n:=n+l; writeln('кол. гласных=',n) end. 2. Заменить в арифметическом выражении функцию sqr на ехр. Замена выражения sqr на ехр достигается последовательным применением процедур delete и insert:
Программа 12 program stroka; var a,b:string[40]; k:integer; begin writeln('введите строку <= 40 символов'); readin(a);b:=a; repeat k:=pos('sqr',b); if k>0 then begin delete(b,k,3);insert('ехр',b,k) ; end until k=0; writein('старая строка=',a); writein('новая строка"',b); end.
3. Ввести и упорядочить по алфавиту 10 латинских слов. В программе определим массивиз 10 элементов-строк и упорядочим его элементы методом пузырька.
Программа 13 program order; const s=10; type word=string(20] ; var i, j, k: 1. . s; b:word; p:boolean; list :array[l..s] of word; begin clrscr;writeln<'введите список слов'); for i:=l to s do readln(list[i]); repeat p:=true; for i:=s downto 2 do if list[i]<list[i-l] then begin b:list[i);list[i]:=list[i-l]; list[i-l]:=b;p:=false end until p=true; writein('упорядоченныйсписок слов:'); for i:=l to s do writeln(list[i]) end. Множество в Паскале имеет такой же смысл, как и в алгебре - это неупорядоченная совокупность отличных друг от друга однотипных элементов. Число элементов множества не должно превышать 255. В качестве типа элементов может быть любой скалярный тип, кроме типа integer и его интервалов, содержащих числа > 255. Тип элементов множества называется базовым. При описании множественного типа используются служебные слова set и of. Задание конкретного множества определяется правилом (конструктором) - списком элементов или интервалов, заключенным в квадратные скобки. Пустое множество обозначается двумя символами []. Множественный тип можно определить в разделе описания типов по схеме:
type <имя> = set of <тип элементов>
Например:
type t=set of byte; var a:t;
Можно совместить описание множественного типа и соответствующих переменных:
var code: set of0..7; digits: set of'0'..'9';
Можно описать переменную множественного типа и задать ее первоначальное значение в разделе описания констант, как константмножество. Тип множества можно описать ранее, например,
type up=setof'A'..'Z'; low=set of'a'.. "z"; const upcase : up=['A'. . 'Z']; vocals :low=['a', 'e', 'i', 'o', "u", 'y']; delimiterset of char=[ '..' /',':'..' ? '];
Для данных множественного типа определены операции объединения, пересечения и дополнения множеств, обозначаемые в Паскале соответственно знаками +, * и -, а также отношения равенства множеств (А=В), неравенства (А<>В), включения (А<=В,А>=В). Логическая операция принадлежностих in А принимает значение true, если элемент х принадлежит множеству А и false в противном случае. Так как к элементам множества прямого доступа нет, то операция in часто используется для этой цели. Заметим, что операции отношения на множествах выполняются быстрее, чем соответствующие операции на числах, поэтому их выгодно применять в программах. Пример: составить программу, анализирующую латинский текст и печатающую в алфавитном порядке все найденные в нем буквы, а затем все ненайденные. Пусть alfa - множество всех букв латинского алфавита. Будем вводить заданный текст с клавиатуры символ за символом, одновременно формируя множество Е -множество латинских букв текста. В конце текста введем символ *. Затем с помощью операции in будем проверять, какие буквы алфавита имеются во множестве Е. Множество N - ненайденных букв в тексте - определяется оператором: N := alfa - Е .
Программа 14 program search; const alfa:set of char=['a'. .'z']; var c:char;E,N:set of char; begin cirscr; E:=[]; writeln('введите текст, конец ввода -*'); read(c); while c<> '*' do begin if с in alfa then E:=E+[c]; read(c) end; writeln; if E=alfa then writeln('найдены все латинские буквы') else begin N:=alfa-E; writeln('найдены:'); for c:='a' to 'z' do if с in E then write(c); writeln; writeln('не найдены:'); for c:='a' to 'z' do if с in N then write(c); writeln end end.
Переменные множественного типа удобно применять в задачах, где порядок данных не имеет значения, например при моделировании случайных событий. Пример: составить программу «спортлото 5 из 36», которая позволяет человеку ввести с клавиатуры пять натуральных чисел из интервала 1..36, затем генерирует случайным образом пять различных чисел из того же интервала и объявляет величину выигрыша по правилу: если угаданы человеком 0, 1 или 2 числа, объявляется проигрыш; если угаданы 3 числа, объявляется выигрыш 3 рубля; если угаданы 4 числа, объявляется выигрыш 100 рублей; если угаданы 5 чисел, объявляется выигрыш 1000 рублей. В программе используются обозначения: mn — множество натуральных чисел из интервала 1 . . 36, а - множество чисел, задуманных человеком, х - множество чисел, . генерируемых компьютером, z=a*x - пересечение множеств а и х; i, k, s - переменные, значения которых принадлежат интервалу 1..36. Случайное число из этого интервала генерируется оператором: s:=random(35)+l. Программа сначала выводит на экран сообщение о выигрышных номерах, затем определяет величину выигрыша.
Программа 15 program lottery; type mn = set of 1 . . 36; var x,a,z: nm; i, k, s: 0 . . 36; begin writeln; a:=[]; for i:=l to 5 do begin write('введите ',i,' -тое число '); readin(k);a:=a+[k] end; randomize; k:=0; x:=[]; whilek<5 do begin s:random (35) +1; if not(s in x) then begin k:=k+l; x:=x+[s] end end; writeln; writeln('выигрыш выпал на следующие номера : '); for i:=l to 36 do if (i in x) then write(i,' '); writeln; z:=a*x; k:=0; for i:=l to 36 do if (i in z) then begin writeln('угадано: ',i); k:=k+l end; case k of 0, 1, 2 : writeln('вы проиграли '); 3 : writeln('получите 3 руб') ; 4 : writeln('получите 100 руб'); 5 : writein('получите 1000 руб') end end. Записи (комбинированный тип) - одна из наиболее гибких и удобных структур данных, применяющихся при описании сложных объектов, которые характеризуются различными свойствами, а также при создании различных информационных систем. Запись - это последовательность, состоящая из фиксированного числа величин разных типов, называемых полями или компонентами записи. Так же, как и массив, запись содержит ряд отдельных компонент, но компонентами записи могут быть данные различных типов. Например, адресные данные (индекс, город, улица, номер дома, квартиры) можно представить как запись (record):
type address = record index : string[6]; city: string[20]; street : string[20]; haus.ilat: integer end;
Из примера видно, что тип «запись» описывается по схеме
type имя типа записи = record имя поля 1 : тип; имя поля 2 : тип; имя поля N: тип end;
Как и при описании массивов, можно совместить описание типа записи и соответствующих переменных. Например, данные о двух студентах можно описать так:
fio :string[20]; fas :string[10] grup :string[8] end
Переменную типа «запись» и ее первоначальное значение можно определить как констант-запись в разделе констант по схеме
const<имя>: <имя типа> = <константное значение>
Константное значение - это список имен полей и соответствующих значений, заключенный в круглые скобки. Элементы списка разделяются знаком «точка с запятой». Например, запись о начале координат можно определить так:
type point = record х, у, z: integer end; const o: point = (x:0; y:0; z:0);
С компонентами записи можно обращаться как с переменными соответствующего типа. Обращение к компонентам записи осуществляется с помощью указания имени поля через точку. Пусть, например, переменная х имеет тип address, т.е. в программе имеется описание var x: address. Тогда допустимы следующие присваивания:
x.haus := 52; х.street:='пр.Мира'; x.city:= 'Красноярск'; x.flat:= 135; x.index :='б60049'
Проиллюстрируем работу с записями на задаче, в которой требуется найти сумму и произведение двух комплексных чисел:
zl=al+i*blиz2=a2+i*b2.
Программа 16 program cornpl; type compi =record re : real; im : real end; var zl,z2,s,p : compl; begin writeln('компл.число a+i*b вводите двумя числами а и Ь: 'it-write('введи 1 число: '); readln(zl.re,zl.im); write('введи 2 число: '); readin(z2.re,z2.im); s.re := zl.re + z2.re; s.im := zl.re + z2.im; p.re := zl.re * z2.re - zl.im * z2.im; p.im := zl.re * z2.im + z2.re * zl.im; writeln('s=',s.re:4:2,' + i *',s.im:4:2); write('p=',p.re:4:2,' + i *',p.im:4:2) end.
Громоздкость обозначений в программе компенсируется большей наглядностью алгоритма за счет структуризации данных. Во многих случаях, если требуется производить операции с полями фиксированной записи, можно для сокращения обозначений использовать оператор присоединения with. Его структура такова:
with <имя записи> do <оператор>;
В этом случае в операторе, написанном после служебного слова do, имена полей указанной записи описываются без имени записи и точки. Например, печать суммы s в предыдущем примере можно организовать с использованием оператора with так:
with s do writeln('s=',re:4:2,'+i*',im:4:2);
В операторах присваивания разрешается использовать не только имена полей, но и имена записей. Тип поля может быть записью. Например:
man = record fio:record fam, im,otch: string[10]; end; data : record day: 1..31; mes:1..12; god:integer end; pol: char; telef: record dom,rab : string[10]; end; end;
В Паскале разрешается использовать тип «запись» при описании другихсоставных типов данных, например, можно построить массив записей. Рассмотрим пример эффективного использования записей в программе начисления стипендии студентам по шаблону:
N ФИО Эк1 Эк2 ЭкЗ Балл Сумма Проф Итого 1 Васнецов Н.В. 4 4 3 11 50.00 0.25 49.75
Предположим, что вводится список группы с соответствующими оценками за экзамены. Графа «Балл» вычисляет суммарную оценку за семестр. Графа «Сумма» определяет размер стипендии по упрощенному правилу: если нет двоек и балл равен 15, то стипендия 75 руб.; при условии, что 12 < «Балл» < 15 стипендия 62 руб 50 коп., а если 9 < «Балл» < 12 , то - 50 руб. (в других случаях сумма равна нулю). В графе «Проф» указывается профсоюзный взнос в размере 0,5% от стипендии, а графа «Итого» определяет сумму денег к выдаче. В программе перед распечаткой итоговой ведомостиможно предусмотреть упорядочение записей по убыванию в графе «Балл».
Программа 17 program spisok; type stud = record fio :string[20]; ex1, ex2, ex3 : 2 . . 5; bal :6 . . 15; sum :real; nalog :real; itog :real; end; var x : array[1..30] of stud; i,k,m,n :integer; у : 6..15; z : stud; begin write('введи число студентов: '); readln(n); for i:= 1 to n do with x[i] do begin write('введи ФИО ',i,'-ro студента: '); readln(fio); write('Bведи его три оценки: '); readin(exl,ex2,ex3); end; for i:= 1 to n do with x[i] do begin bal:=exl+ex2+ex3; if (exl=2) or (ex2=2) or (ex3=2) then sum:=0 else if bal=15 then sum:=75 else if bal>12 then sum:=62.5 else if bal>9 then sum:=50 else sum:=0; nalog:=sum*0.005; itog:=sum-nalog; end; for k:= 1 to n-1 do begin y:=x[k].bal; m:=k; for i:=k+l to n do if y<x[i].bal then begin y:=x[i].bal; m:=i end; z:=x[k]; x(k]:=x[m]; x(m]:=z; end; writeln; writeln ('СТИПЕНДИАЛЬНАЯ ВЕДОМОСТЬ '); for i:=l to 64 do write('-'); writeln; write ('N | ФИО | эк1 | эк2 | эк3 | балл | сумма | проф | итого |') ; for i:=l to 64 do write('-'); writeln; for i:=l to n do with x[i] do begin write(i:3,fio:20,exi:4, ex2:4,ex3:4); writeln(bal:5,sum:9:2,nalog:8:2,- itog:7:2); end end.
Контрольные вопросы и задания
1. Как определяется перечислимый тип данных? 2. Для чего может понадобиться интервальный тип данных? 3. Как вводятся и используются в программах массивы? 4. Какие действия возможны над величинами строкового типа? 5. Какие операции допустимы над множествами? 6. В чем принципиальные различия между одномерными массивами и записями? 7. В усеченном конусе длина диагонали осевого сечения равна d, образующая составляет с плоскостью основания угол х и равна а. Вычислите площадь боковой поверхности конуса. 8. Вычислите объем призмы, боковые грани которой квадраты, а основанием служит равносторонний треугольник, вписанный в круг радиуса г. 9. Числа а и b выражают длины катетов одного прямоугольного треугольника, числа с и d - другого. Определите, являются ли треугольники подобными. 10. Напечатайте числа а, b, с в порядке возрастания. 11. Определите все пары двузначных чисел, обладающих свойством: (20+25^2 = = 2025. 12. Вычислите в числовом массиве а1,а2,...,аn суммы положительных и отрицательных элементов. 13. Вычислите скалярное произведение двух десятимерных векторов Х и Y. 14. Упорядочите массив х1,х2,...,хn по неубыванию, используя метод сортировки вставками: пусть первые k элементов уже упорядочены по неубыванию; берется (K+1)-й элемент и размещается среди первых k элементов так, чтобы упорядоченными оказались уже (k+1) первых элементов. 15. Составьте программу решения треугольной системы уравнений порядка n. 16. Замените в заданном арифметическом выражении все вхождения sin на cos и sqrt на abs. 17. Для заданного текста определите длину содержащейся в нем максимальной последовательности цифр 0, 1,2,..., 9. 18. Дан текст из латинских букв и знаков препинания. Составьте программу частотного анализа букв этого текста, т.е. напечатайте каждую букву с указанием количества ее вхождений и процента вхождений. 19. Найдите и напечатайте в порядке убывания все простые числа из диапазона [2..201]. 20. Опишите тип «запись» для следующих данных: а) адрес(город, улица, дом, квартира); б) дата(число,месяц.год); в) студент(фио,факультет,курс,группа). 21. Заданы N точек на плоскости. Найдите точку, ближайшую к началу координат. Используйте тип «запись».
ПРОЦЕДУРЫ И ФУНКЦИИ
Описание и вызов. В Паскале подпрограммы называются процедурами и функциями и описываются в разделе с тем же названием. Процедура имеет такую же структуру, как и программа, но с двумя отличиями: • заголовок процедуры имеет другой синтаксис и включает служебное слово procedure; • описание процедуры заканчивается точкой с запятой (а не точкой). Все имена, описанные в программе до процедуры, действуют во всей программе и в любой ее подпрограмме (если они там не описаны заново). Они называются глобальными, в отличие от локальных имен, описанных в процедуре и действующих лишь в ней. Данные для обработки могут передаваться процедуре через глобальные имена или через аргументы процедуры. В процедуре каждый аргумент имеет свое имя -формальный параметр, описываемый в заголовке процедуры по схеме
procedure <имя> (<список описаний формальных параметров>) Описание формальных параметров может иметь вид <список имен>: <тип> или var <список имен>: <тип>
В первом случае говорят о параметрах-значениях, во втором - о параметрах-переменных. В простейшем случае заголовок процедуры может содержать только имя процедуры. Оператор вызова процедуры имеет вид
<имя процедуры> (<список выражений>);
Указанные выражения называют фактическими параметрами. Их список должен точно соответствовать списку описаний формальных параметров процедуры. Во время вызова процедуры каждому параметру-значению присваивается значение соответствующего фактического параметра и поэтому их обычно используют для передачи входных данных. Параметры-переменные следует использовать для представления результатов процедуры. Пример: составим программу, которая с помощью строки символов разделит экран на части, где напечатает таблицу квадратных корней для чисел 1, 2,..., 10 и таблицу натуральных логарифмов для чисел 1, 2,..., 5. Печатьстроки символов оформим как процедуру. Так как никакую информацию передавать из процедуры в программу не надо, то аргументы процедуры (вид и количество символов) будут описаны как параметры-значения. Заметим, что процедура в программе выполняется пять раз.
Программа 18 program section; var x:integer; procedure line(a:integer;c:char) ; var j:integer; begin for j:=l to a do write (c); writeln end; begin line(35,'-'); writeln('таблица квадратных корней'); line(35,'-'); for x:=l to 10 do writeln(x:8,sqrt(x):8,4); line (35,'-'); writein('таблица натуральных логарифмов'); line(35,'-'); for x:=l to 5 do writein(x:8,In(x):8:4); line(35,'*') end. Функция - это подпрограмма, определяющая единственное скалярное, вещественное или строковое значение. Отличия подпрограммы-функции от процедуры: • заголовок функции начинается со служебного слова function и заканчивается указанием типа значения функции:
function <имя> (список описаний формальных параметров): <тип>;
•раздел операторов функции должен содержать хотя бы один оператор присваивания имени функции; • обращение к функции - не оператор, а выражение вида
<имя функции> (<список фактических параметров>).
Функции (и процедуры) могут использовать свое имя в собственном описании, т.е. могут быть рекурсивными. Пример: составим программу, которая для заданных четырех натуральных чисел а, b, с, d напечатает наибольшие общие делители первой и второй пар чисел и сравнит их по величине. В программе определим рекурсивную функцию nod(x,y) по формулам
| x, если у = 0 |
||
Последнее изменение этой страницы: 2018-05-10; просмотров: 353. stydopedya.ru не претендует на авторское право материалов, которые вылажены, но предоставляет бесплатный доступ к ним. В случае нарушения авторского права или персональных данных напишите сюда... |